Removed custom ESLint curly rule

Sigh, why did you do this, past me..?

Also fixed "Delete uploads by names".
This commit is contained in:
Bobby Wibowo 2020-10-31 01:12:09 +07:00
parent a86967c251
commit 47dd512910
No known key found for this signature in database
GPG Key ID: 51C3A1E1E22D26CF
20 changed files with 902 additions and 737 deletions

View File

@ -10,16 +10,11 @@ module.exports = {
'standard' 'standard'
], ],
rules: { rules: {
curly: [
'error',
'multi',
'consistent'
],
'no-throw-literal': 0, 'no-throw-literal': 0,
'object-shorthand': [ 'object-shorthand': [
'error', 'error',
'always' 'always'
], ],
'standard/no-callback-literal': 0 'node/no-callback-literal': 0
} }
} }

View File

@ -29,14 +29,10 @@ const zipOptions = config.uploads.jsZipOptions
zipOptions.type = 'nodebuffer' zipOptions.type = 'nodebuffer'
// Apply fallbacks for missing config values // Apply fallbacks for missing config values
if (zipOptions.streamFiles === undefined) if (zipOptions.streamFiles === undefined) zipOptions.streamFiles = true
zipOptions.streamFiles = true if (zipOptions.compression === undefined) zipOptions.compression = 'DEFLATE'
if (zipOptions.compression === undefined) if (zipOptions.compressionOptions === undefined) zipOptions.compressionOptions = {}
zipOptions.compression = 'DEFLATE' if (zipOptions.compressionOptions.level === undefined) zipOptions.compressionOptions.level = 1
if (zipOptions.compressionOptions === undefined)
zipOptions.compressionOptions = {}
if (zipOptions.compressionOptions.level === undefined)
zipOptions.compressionOptions.level = 1
self.zipEmitters = new Map() self.zipEmitters = new Map()
@ -51,8 +47,7 @@ class ZipEmitter extends EventEmitter {
self.getUniqueRandomName = async () => { self.getUniqueRandomName = async () => {
for (let i = 0; i < utils.idMaxTries; i++) { for (let i = 0; i < utils.idMaxTries; i++) {
const identifier = randomstring.generate(config.uploads.albumIdentifierLength) const identifier = randomstring.generate(config.uploads.albumIdentifierLength)
if (self.onHold.has(identifier)) if (self.onHold.has(identifier)) continue
continue
// Put token on-hold (wait for it to be inserted to DB) // Put token on-hold (wait for it to be inserted to DB)
self.onHold.add(identifier) self.onHold.add(identifier)
@ -80,15 +75,15 @@ self.list = async (req, res, next) => {
const all = req.headers.all === '1' const all = req.headers.all === '1'
const sidebar = req.headers.sidebar const sidebar = req.headers.sidebar
const ismoderator = perms.is(user, 'moderator') const ismoderator = perms.is(user, 'moderator')
if (all && !ismoderator) if (all && !ismoderator) return res.status(403).end()
return res.status(403).end()
const filter = function () { const filter = function () {
if (!all) if (!all) {
this.where({ this.where({
enabled: 1, enabled: 1,
userid: user.id userid: user.id
}) })
}
} }
try { try {
@ -97,8 +92,7 @@ self.list = async (req, res, next) => {
.where(filter) .where(filter)
.count('id as count') .count('id as count')
.then(rows => rows[0].count) .then(rows => rows[0].count)
if (!count) if (!count) return res.json({ success: true, albums: [], count })
return res.json({ success: true, albums: [], count })
const fields = ['id', 'name'] const fields = ['id', 'name']
@ -116,8 +110,7 @@ self.list = async (req, res, next) => {
else if (offset < 0) offset = Math.max(0, Math.ceil(count / 25) + offset) else if (offset < 0) offset = Math.max(0, Math.ceil(count / 25) + offset)
fields.push('identifier', 'enabled', 'timestamp', 'editedAt', 'download', 'public', 'description') fields.push('identifier', 'enabled', 'timestamp', 'editedAt', 'download', 'public', 'description')
if (all) if (all) fields.push('userid')
fields.push('userid')
albums = await db.table('albums') albums = await db.table('albums')
.where(filter) .where(filter)
@ -140,13 +133,14 @@ self.list = async (req, res, next) => {
.whereIn('albumid', Object.keys(albumids)) .whereIn('albumid', Object.keys(albumids))
.select('albumid') .select('albumid')
for (const upload of uploads) for (const upload of uploads) {
if (albumids[upload.albumid]) if (albumids[upload.albumid]) {
albumids[upload.albumid].uploads++ albumids[upload.albumid].uploads++
}
}
// If we are not listing all albums, send response // If we are not listing all albums, send response
if (!all) if (!all) return res.json({ success: true, albums, count, homeDomain })
return res.json({ success: true, albums, count, homeDomain })
// Otherwise proceed to querying usernames // Otherwise proceed to querying usernames
const userids = albums const userids = albums
@ -156,8 +150,7 @@ self.list = async (req, res, next) => {
}) })
// If there are no albums attached to a registered user, send response // If there are no albums attached to a registered user, send response
if (userids.length === 0) if (!userids.length) return res.json({ success: true, albums, count, homeDomain })
return res.json({ success: true, albums, count, homeDomain })
// Query usernames of user IDs from currently selected files // Query usernames of user IDs from currently selected files
const usersTable = await db.table('users') const usersTable = await db.table('users')
@ -165,8 +158,9 @@ self.list = async (req, res, next) => {
.select('id', 'username') .select('id', 'username')
const users = {} const users = {}
for (const user of usersTable) for (const user of usersTable) {
users[user.id] = user.username users[user.id] = user.username
}
return res.json({ success: true, albums, count, users, homeDomain }) return res.json({ success: true, albums, count, users, homeDomain })
} catch (error) { } catch (error) {
@ -183,8 +177,7 @@ self.create = async (req, res, next) => {
? utils.escape(req.body.name.trim().substring(0, self.titleMaxLength)) ? utils.escape(req.body.name.trim().substring(0, self.titleMaxLength))
: '' : ''
if (!name) if (!name) return res.json({ success: false, description: 'No album name specified.' })
return res.json({ success: false, description: 'No album name specified.' })
try { try {
const album = await db.table('albums') const album = await db.table('albums')
@ -195,8 +188,7 @@ self.create = async (req, res, next) => {
}) })
.first() .first()
if (album) if (album) return res.json({ success: false, description: 'There is already an album with that name.' })
return res.json({ success: false, description: 'There is already an album with that name.' })
const identifier = await self.getUniqueRandomName() const identifier = await self.getUniqueRandomName()
@ -235,8 +227,7 @@ self.disable = async (req, res, next) => {
const id = req.body.id const id = req.body.id
const purge = req.body.purge const purge = req.body.purge
if (!Number.isFinite(id)) if (!Number.isFinite(id)) return res.json({ success: false, description: 'No album specified.' })
return res.json({ success: false, description: 'No album specified.' })
try { try {
if (purge) { if (purge) {
@ -249,8 +240,7 @@ self.disable = async (req, res, next) => {
if (files.length) { if (files.length) {
const ids = files.map(file => file.id) const ids = files.map(file => file.id)
const failed = await utils.bulkDeleteFromDb('id', ids, user) const failed = await utils.bulkDeleteFromDb('id', ids, user)
if (failed.length) if (failed.length) return res.json({ success: false, failed })
return res.json({ success: false, failed })
} }
utils.invalidateStatsCache('uploads') utils.invalidateStatsCache('uploads')
} }
@ -291,24 +281,23 @@ self.edit = async (req, res, next) => {
const ismoderator = perms.is(user, 'moderator') const ismoderator = perms.is(user, 'moderator')
const id = parseInt(req.body.id) const id = parseInt(req.body.id)
if (isNaN(id)) if (isNaN(id)) return res.json({ success: false, description: 'No album specified.' })
return res.json({ success: false, description: 'No album specified.' })
const name = typeof req.body.name === 'string' const name = typeof req.body.name === 'string'
? utils.escape(req.body.name.trim().substring(0, self.titleMaxLength)) ? utils.escape(req.body.name.trim().substring(0, self.titleMaxLength))
: '' : ''
if (!name) if (!name) return res.json({ success: false, description: 'No name specified.' })
return res.json({ success: false, description: 'No name specified.' })
const filter = function () { const filter = function () {
this.where('id', id) this.where('id', id)
if (!ismoderator) if (!ismoderator) {
this.andWhere({ this.andWhere({
enabled: 1, enabled: 1,
userid: user.id userid: user.id
}) })
}
} }
try { try {
@ -316,13 +305,14 @@ self.edit = async (req, res, next) => {
.where(filter) .where(filter)
.first() .first()
if (!album) if (!album) {
return res.json({ success: false, description: 'Could not get album with the specified ID.' }) return res.json({ success: false, description: 'Could not get album with the specified ID.' })
else if (album.id !== id) } else if (album.id !== id) {
return res.json({ success: false, description: 'Name already in use.' }) return res.json({ success: false, description: 'Name already in use.' })
else if (req._old && (album.id === id)) } else if (req._old && (album.id === id)) {
// Old rename API // Old rename API
return res.json({ success: false, description: 'You did not specify a new name.' }) return res.json({ success: false, description: 'You did not specify a new name.' })
}
const update = { const update = {
name, name,
@ -333,11 +323,13 @@ self.edit = async (req, res, next) => {
: '' : ''
} }
if (ismoderator) if (ismoderator) {
update.enabled = Boolean(req.body.enabled) update.enabled = Boolean(req.body.enabled)
}
if (req.body.requestLink) if (req.body.requestLink) {
update.identifier = await self.getUniqueRandomName() update.identifier = await self.getUniqueRandomName()
}
await db.table('albums') await db.table('albums')
.where(filter) .where(filter)
@ -353,10 +345,9 @@ self.edit = async (req, res, next) => {
const oldZip = path.join(paths.zips, `${album.identifier}.zip`) const oldZip = path.join(paths.zips, `${album.identifier}.zip`)
const newZip = path.join(paths.zips, `${update.identifier}.zip`) const newZip = path.join(paths.zips, `${update.identifier}.zip`)
await paths.rename(oldZip, newZip) await paths.rename(oldZip, newZip)
} catch (err) { } catch (error) {
// Re-throw error // Re-throw error
if (err.code !== 'ENOENT') if (error.code !== 'ENOENT') throw error
throw err
} }
return res.json({ return res.json({
@ -380,8 +371,9 @@ self.rename = async (req, res, next) => {
self.get = async (req, res, next) => { self.get = async (req, res, next) => {
const identifier = req.params.identifier const identifier = req.params.identifier
if (identifier === undefined) if (identifier === undefined) {
return res.status(401).json({ success: false, description: 'No identifier provided.' }) return res.status(401).json({ success: false, description: 'No identifier provided.' })
}
try { try {
const album = await db.table('albums') const album = await db.table('albums')
@ -391,16 +383,17 @@ self.get = async (req, res, next) => {
}) })
.first() .first()
if (!album) if (!album) {
return res.json({ return res.json({
success: false, success: false,
description: 'Album not found.' description: 'Album not found.'
}) })
else if (album.public === 0) } else if (album.public === 0) {
return res.status(403).json({ return res.status(403).json({
success: false, success: false,
description: 'This album is not available for public.' description: 'This album is not available for public.'
}) })
}
const title = album.name const title = album.name
const files = await db.table('files') const files = await db.table('files')
@ -412,8 +405,9 @@ self.get = async (req, res, next) => {
file.file = `${config.domain}/${file.name}` file.file = `${config.domain}/${file.name}`
const extname = utils.extname(file.name) const extname = utils.extname(file.name)
if (utils.mayGenerateThumb(extname)) if (utils.mayGenerateThumb(extname)) {
file.thumb = `${config.domain}/thumbs/${file.name.slice(0, -extname.length)}.png` file.thumb = `${config.domain}/thumbs/${file.name.slice(0, -extname.length)}.png`
}
} }
return res.json({ return res.json({
@ -432,17 +426,19 @@ self.generateZip = async (req, res, next) => {
const versionString = parseInt(req.query.v) const versionString = parseInt(req.query.v)
const identifier = req.params.identifier const identifier = req.params.identifier
if (identifier === undefined) if (identifier === undefined) {
return res.status(401).json({ return res.status(401).json({
success: false, success: false,
description: 'No identifier provided.' description: 'No identifier provided.'
}) })
}
if (!config.uploads.generateZips) if (!config.uploads.generateZips) {
return res.status(401).json({ return res.status(401).json({
success: false, success: false,
description: 'Zip generation disabled.' description: 'Zip generation disabled.'
}) })
}
try { try {
const album = await db.table('albums') const album = await db.table('albums')
@ -452,32 +448,35 @@ self.generateZip = async (req, res, next) => {
}) })
.first() .first()
if (!album) if (!album) {
return res.json({ success: false, description: 'Album not found.' }) return res.json({ success: false, description: 'Album not found.' })
else if (album.download === 0) } else if (album.download === 0) {
return res.json({ success: false, description: 'Download for this album is disabled.' }) return res.json({ success: false, description: 'Download for this album is disabled.' })
}
if ((isNaN(versionString) || versionString <= 0) && album.editedAt) if ((isNaN(versionString) || versionString <= 0) && album.editedAt) {
return res.redirect(`${album.identifier}?v=${album.editedAt}`) return res.redirect(`${album.identifier}?v=${album.editedAt}`)
}
if (album.zipGeneratedAt > album.editedAt) if (album.zipGeneratedAt > album.editedAt) {
try { try {
const filePath = path.join(paths.zips, `${identifier}.zip`) const filePath = path.join(paths.zips, `${identifier}.zip`)
await paths.access(filePath) await paths.access(filePath)
return res.download(filePath, `${album.name}.zip`) return res.download(filePath, `${album.name}.zip`)
} catch (error) { } catch (error) {
// Re-throw error // Re-throw error
if (error.code !== 'ENOENT') if (error.code !== 'ENOENT') throw error
throw error
} }
}
if (self.zipEmitters.has(identifier)) { if (self.zipEmitters.has(identifier)) {
logger.log(`Waiting previous zip task for album: ${identifier}.`) logger.log(`Waiting previous zip task for album: ${identifier}.`)
return self.zipEmitters.get(identifier).once('done', (filePath, fileName, json) => { return self.zipEmitters.get(identifier).once('done', (filePath, fileName, json) => {
if (filePath && fileName) if (filePath && fileName) {
res.download(filePath, fileName) res.download(filePath, fileName)
else if (json) } else if (json) {
res.json(json) res.json(json)
}
}) })
} }
@ -559,8 +558,9 @@ self.addFiles = async (req, res, next) => {
if (!user) return if (!user) return
const ids = req.body.ids const ids = req.body.ids
if (!Array.isArray(ids) || !ids.length) if (!Array.isArray(ids) || !ids.length) {
return res.json({ success: false, description: 'No files specified.' }) return res.json({ success: false, description: 'No files specified.' })
}
let albumid = parseInt(req.body.albumid) let albumid = parseInt(req.body.albumid)
if (isNaN(albumid) || albumid < 0) albumid = null if (isNaN(albumid) || albumid < 0) albumid = null
@ -572,16 +572,18 @@ self.addFiles = async (req, res, next) => {
const album = await db.table('albums') const album = await db.table('albums')
.where('id', albumid) .where('id', albumid)
.where(function () { .where(function () {
if (user.username !== 'root') if (user.username !== 'root') {
this.where('userid', user.id) this.where('userid', user.id)
}
}) })
.first() .first()
if (!album) if (!album) {
return res.json({ return res.json({
success: false, success: false,
description: 'Album does not exist or it does not belong to the user.' description: 'Album does not exist or it does not belong to the user.'
}) })
}
albumids.push(albumid) albumids.push(albumid)
} }
@ -597,8 +599,9 @@ self.addFiles = async (req, res, next) => {
.update('albumid', albumid) .update('albumid', albumid)
files.forEach(file => { files.forEach(file => {
if (file.albumid && !albumids.includes(file.albumid)) if (file.albumid && !albumids.includes(file.albumid)) {
albumids.push(file.albumid) albumids.push(file.albumid)
}
}) })
await db.table('albums') await db.table('albums')
@ -609,13 +612,14 @@ self.addFiles = async (req, res, next) => {
return res.json({ success: true, failed }) return res.json({ success: true, failed })
} catch (error) { } catch (error) {
logger.error(error) logger.error(error)
if (failed.length === ids.length) if (failed.length === ids.length) {
return res.json({ return res.json({
success: false, success: false,
description: `Could not ${albumid === null ? 'add' : 'remove'} any files ${albumid === null ? 'to' : 'from'} the album.` description: `Could not ${albumid === null ? 'add' : 'remove'} any files ${albumid === null ? 'to' : 'from'} the album.`
}) })
else } else {
return res.status(500).json({ success: false, description: 'An unexpected error occurred. Try again?' }) return res.status(500).json({ success: false, description: 'An unexpected error occurred. Try again?' })
}
} }
} }

View File

@ -34,31 +34,30 @@ self.verify = async (req, res, next) => {
const username = typeof req.body.username === 'string' const username = typeof req.body.username === 'string'
? req.body.username.trim() ? req.body.username.trim()
: '' : ''
if (!username) if (!username) return res.json({ success: false, description: 'No username provided.' })
return res.json({ success: false, description: 'No username provided.' })
const password = typeof req.body.password === 'string' const password = typeof req.body.password === 'string'
? req.body.password.trim() ? req.body.password.trim()
: '' : ''
if (!password) if (!password) return res.json({ success: false, description: 'No password provided.' })
return res.json({ success: false, description: 'No password provided.' })
try { try {
const user = await db.table('users') const user = await db.table('users')
.where('username', username) .where('username', username)
.first() .first()
if (!user) if (!user) return res.json({ success: false, description: 'Username does not exist.' })
return res.json({ success: false, description: 'Username does not exist.' })
if (user.enabled === false || user.enabled === 0) if (user.enabled === false || user.enabled === 0) {
return res.json({ success: false, description: 'This account has been disabled.' }) return res.json({ success: false, description: 'This account has been disabled.' })
}
const result = await bcrypt.compare(password, user.password) const result = await bcrypt.compare(password, user.password)
if (result === false) if (result === false) {
return res.json({ success: false, description: 'Wrong password.' }) return res.json({ success: false, description: 'Wrong password.' })
else } else {
return res.json({ success: true, token: user.token }) return res.json({ success: true, token: user.token })
}
} catch (error) { } catch (error) {
logger.error(error) logger.error(error)
return res.status(500).json({ success: false, description: 'An unexpected error occurred. Try again?' }) return res.status(500).json({ success: false, description: 'An unexpected error occurred. Try again?' })
@ -66,34 +65,46 @@ self.verify = async (req, res, next) => {
} }
self.register = async (req, res, next) => { self.register = async (req, res, next) => {
if (config.enableUserAccounts === false) if (config.enableUserAccounts === false) {
return res.json({ success: false, description: 'Registration is currently disabled.' }) return res.json({ success: false, description: 'Registration is currently disabled.' })
}
const username = typeof req.body.username === 'string' const username = typeof req.body.username === 'string'
? req.body.username.trim() ? req.body.username.trim()
: '' : ''
if (username.length < self.user.min || username.length > self.user.max) if (username.length < self.user.min || username.length > self.user.max) {
return res.json({ success: false, description: `Username must have ${self.user.min}-${self.user.max} characters.` }) return res.json({
success: false,
description: `Username must have ${self.user.min}-${self.user.max} characters.`
})
}
const password = typeof req.body.password === 'string' const password = typeof req.body.password === 'string'
? req.body.password.trim() ? req.body.password.trim()
: '' : ''
if (password.length < self.pass.min || password.length > self.pass.max) if (password.length < self.pass.min || password.length > self.pass.max) {
return res.json({ success: false, description: `Password must have ${self.pass.min}-${self.pass.max} characters.` }) return res.json({
success: false,
description: `Password must have ${self.pass.min}-${self.pass.max} characters.`
})
}
try { try {
const user = await db.table('users') const user = await db.table('users')
.where('username', username) .where('username', username)
.first() .first()
if (user) if (user) return res.json({ success: false, description: 'Username already exists.' })
return res.json({ success: false, description: 'Username already exists.' })
const hash = await bcrypt.hash(password, saltRounds) const hash = await bcrypt.hash(password, saltRounds)
const token = await tokens.generateUniqueToken() const token = await tokens.generateUniqueToken()
if (!token) if (!token) {
return res.json({ success: false, description: 'Sorry, we could not allocate a unique token. Try again?' }) return res.json({
success: false,
description: 'Sorry, we could not allocate a unique token. Try again?'
})
}
await db.table('users') await db.table('users')
.insert({ .insert({
@ -121,8 +132,12 @@ self.changePassword = async (req, res, next) => {
const password = typeof req.body.password === 'string' const password = typeof req.body.password === 'string'
? req.body.password.trim() ? req.body.password.trim()
: '' : ''
if (password.length < self.pass.min || password.length > self.pass.max) if (password.length < self.pass.min || password.length > self.pass.max) {
return res.json({ success: false, description: `Password must have ${self.pass.min}-${self.pass.max} characters.` }) return res.json({
success: false,
description: `Password must have ${self.pass.min}-${self.pass.max} characters.`
})
}
try { try {
const hash = await bcrypt.hash(password, saltRounds) const hash = await bcrypt.hash(password, saltRounds)
@ -139,12 +154,13 @@ self.changePassword = async (req, res, next) => {
} }
self.assertPermission = (user, target) => { self.assertPermission = (user, target) => {
if (!target) if (!target) {
throw new Error('Could not get user with the specified ID.') throw new Error('Could not get user with the specified ID.')
else if (!perms.higher(user, target)) } else if (!perms.higher(user, target)) {
throw new Error('The user is in the same or higher group as you.') throw new Error('The user is in the same or higher group as you.')
else if (target.username === 'root') } else if (target.username === 'root') {
throw new Error('Root user may not be tampered with.') throw new Error('Root user may not be tampered with.')
}
} }
self.createUser = async (req, res, next) => { self.createUser = async (req, res, next) => {
@ -152,21 +168,28 @@ self.createUser = async (req, res, next) => {
if (!user) return if (!user) return
const isadmin = perms.is(user, 'admin') const isadmin = perms.is(user, 'admin')
if (!isadmin) if (!isadmin) return res.status(403).end()
return res.status(403).end()
const username = typeof req.body.username === 'string' const username = typeof req.body.username === 'string'
? req.body.username.trim() ? req.body.username.trim()
: '' : ''
if (username.length < self.user.min || username.length > self.user.max) if (username.length < self.user.min || username.length > self.user.max) {
return res.json({ success: false, description: `Username must have ${self.user.min}-${self.user.max} characters.` }) return res.json({
success: false,
description: `Username must have ${self.user.min}-${self.user.max} characters.`
})
}
let password = typeof req.body.password === 'string' let password = typeof req.body.password === 'string'
? req.body.password.trim() ? req.body.password.trim()
: '' : ''
if (password.length) { if (password.length) {
if (password.length < self.pass.min || password.length > self.pass.max) if (password.length < self.pass.min || password.length > self.pass.max) {
return res.json({ success: false, description: `Password must have ${self.pass.min}-${self.pass.max} characters.` }) return res.json({
success: false,
description: `Password must have ${self.pass.min}-${self.pass.max} characters.`
})
}
} else { } else {
password = randomstring.generate(self.pass.rand) password = randomstring.generate(self.pass.rand)
} }
@ -186,14 +209,17 @@ self.createUser = async (req, res, next) => {
.where('username', username) .where('username', username)
.first() .first()
if (user) if (user) return res.json({ success: false, description: 'Username already exists.' })
return res.json({ success: false, description: 'Username already exists.' })
const hash = await bcrypt.hash(password, saltRounds) const hash = await bcrypt.hash(password, saltRounds)
const token = await tokens.generateUniqueToken() const token = await tokens.generateUniqueToken()
if (!token) if (!token) {
return res.json({ success: false, description: 'Sorry, we could not allocate a unique token. Try again?' }) return res.json({
success: false,
description: 'Sorry, we could not allocate a unique token. Try again?'
})
}
await db.table('users') await db.table('users')
.insert({ .insert({
@ -219,12 +245,10 @@ self.editUser = async (req, res, next) => {
if (!user) return if (!user) return
const isadmin = perms.is(user, 'admin') const isadmin = perms.is(user, 'admin')
if (!isadmin) if (!isadmin) return res.status(403).end()
return res.status(403).end()
const id = parseInt(req.body.id) const id = parseInt(req.body.id)
if (isNaN(id)) if (isNaN(id)) return res.json({ success: false, description: 'No user specified.' })
return res.json({ success: false, description: 'No user specified.' })
try { try {
const target = await db.table('users') const target = await db.table('users')
@ -236,17 +260,20 @@ self.editUser = async (req, res, next) => {
if (req.body.username !== undefined) { if (req.body.username !== undefined) {
update.username = String(req.body.username).trim() update.username = String(req.body.username).trim()
if (update.username.length < self.user.min || update.username.length > self.user.max) if (update.username.length < self.user.min || update.username.length > self.user.max) {
throw new Error(`Username must have ${self.user.min}-${self.user.max} characters.`) throw new Error(`Username must have ${self.user.min}-${self.user.max} characters.`)
}
} }
if (req.body.enabled !== undefined) if (req.body.enabled !== undefined) {
update.enabled = Boolean(req.body.enabled) update.enabled = Boolean(req.body.enabled)
}
if (req.body.group !== undefined) { if (req.body.group !== undefined) {
update.permission = perms.permissions[req.body.group] update.permission = perms.permissions[req.body.group]
if (typeof update.permission !== 'number' || update.permission < 0) if (typeof update.permission !== 'number' || update.permission < 0) {
update.permission = target.permission update.permission = target.permission
}
} }
let password let password
@ -282,13 +309,11 @@ self.deleteUser = async (req, res, next) => {
if (!user) return if (!user) return
const isadmin = perms.is(user, 'admin') const isadmin = perms.is(user, 'admin')
if (!isadmin) if (!isadmin) return res.status(403).end()
return res.status(403).end()
const id = parseInt(req.body.id) const id = parseInt(req.body.id)
const purge = req.body.purge const purge = req.body.purge
if (isNaN(id)) if (isNaN(id)) return res.json({ success: false, description: 'No user specified.' })
return res.json({ success: false, description: 'No user specified.' })
try { try {
const target = await db.table('users') const target = await db.table('users')
@ -304,8 +329,7 @@ self.deleteUser = async (req, res, next) => {
const fileids = files.map(file => file.id) const fileids = files.map(file => file.id)
if (purge) { if (purge) {
const failed = await utils.bulkDeleteFromDb('id', fileids, user) const failed = await utils.bulkDeleteFromDb('id', fileids, user)
if (failed.length) if (failed.length) return res.json({ success: false, failed })
return res.json({ success: false, failed })
utils.invalidateStatsCache('uploads') utils.invalidateStatsCache('uploads')
} else { } else {
// Clear out userid attribute from the files // Clear out userid attribute from the files
@ -315,7 +339,8 @@ self.deleteUser = async (req, res, next) => {
} }
} }
// TODO: Figure out obstacles of just deleting the albums // TODO: Figure out why can't we just just delete the albums from DB
// DISCLAIMER: Upstream always had it coded this way for some reason
const albums = await db.table('albums') const albums = await db.table('albums')
.where('userid', id) .where('userid', id)
.where('enabled', 1) .where('enabled', 1)
@ -333,8 +358,7 @@ self.deleteUser = async (req, res, next) => {
try { try {
await paths.unlink(path.join(paths.zips, `${album.identifier}.zip`)) await paths.unlink(path.join(paths.zips, `${album.identifier}.zip`))
} catch (error) { } catch (error) {
if (error.code !== 'ENOENT') if (error.code !== 'ENOENT') throw error
throw error
} }
})) }))
} }
@ -362,15 +386,13 @@ self.listUsers = async (req, res, next) => {
if (!user) return if (!user) return
const isadmin = perms.is(user, 'admin') const isadmin = perms.is(user, 'admin')
if (!isadmin) if (!isadmin) return res.status(403).end()
return res.status(403).end()
try { try {
const count = await db.table('users') const count = await db.table('users')
.count('id as count') .count('id as count')
.then(rows => rows[0].count) .then(rows => rows[0].count)
if (!count) if (!count) return res.json({ success: true, users: [], count })
return res.json({ success: true, users: [], count })
let offset = Number(req.params.page) let offset = Number(req.params.page)
if (isNaN(offset)) offset = 0 if (isNaN(offset)) offset = 0

View File

@ -36,8 +36,9 @@ DiskStorage.prototype._handleFile = function _handleFile (req, file, cb) {
file._chunksData.stream = fs.createWriteStream(finalPath, { flags: 'a' }) file._chunksData.stream = fs.createWriteStream(finalPath, { flags: 'a' })
file._chunksData.stream.on('error', onerror) file._chunksData.stream.on('error', onerror)
} }
if (!file._chunksData.hasher) if (!file._chunksData.hasher) {
file._chunksData.hasher = blake3.createHash() file._chunksData.hasher = blake3.createHash()
}
outStream = file._chunksData.stream outStream = file._chunksData.stream
hash = file._chunksData.hasher hash = file._chunksData.hasher

View File

@ -20,8 +20,9 @@ const fsFuncs = [
'writeFile' 'writeFile'
] ]
for (const fsFunc of fsFuncs) for (const fsFunc of fsFuncs) {
self[fsFunc] = promisify(fs[fsFunc]) self[fsFunc] = promisify(fs[fsFunc])
}
self.uploads = path.resolve(config.uploads.folder) self.uploads = path.resolve(config.uploads.folder)
self.chunks = path.join(self.uploads, 'chunks') self.chunks = path.join(self.uploads, 'chunks')
@ -51,7 +52,7 @@ const verify = [
self.init = async () => { self.init = async () => {
// Check & create directories // Check & create directories
for (const p of verify) for (const p of verify) {
try { try {
await self.access(p) await self.access(p)
} catch (err) { } catch (err) {
@ -59,10 +60,10 @@ self.init = async () => {
throw err throw err
} else { } else {
const mkdir = await self.mkdir(p) const mkdir = await self.mkdir(p)
if (mkdir) if (mkdir) logger.log(`Created directory: ${p}`)
logger.log(`Created directory: ${p}`)
} }
} }
}
// Purge any leftover in chunks directory // Purge any leftover in chunks directory
const uuidDirs = await self.readdir(self.chunks) const uuidDirs = await self.readdir(self.chunks)
@ -74,8 +75,7 @@ self.init = async () => {
)) ))
await self.rmdir(root) await self.rmdir(root)
})) }))
if (uuidDirs.length) if (uuidDirs.length) logger.log(`Purged ${uuidDirs.length} unfinished chunks`)
logger.log(`Purged ${uuidDirs.length} unfinished chunks`)
} }
module.exports = self module.exports = self

View File

@ -11,8 +11,7 @@ self.permissions = {
// returns true if user is in the group OR higher // returns true if user is in the group OR higher
self.is = (user, group) => { self.is = (user, group) => {
// root bypass // root bypass
if (user.username === 'root') if (user.username === 'root') return true
return true
const permission = user.permission || 0 const permission = user.permission || 0
return permission >= self.permissions[group] return permission >= self.permissions[group]

View File

@ -14,8 +14,7 @@ const self = {
self.generateUniqueToken = async () => { self.generateUniqueToken = async () => {
for (let i = 0; i < self.tokenMaxTries; i++) { for (let i = 0; i < self.tokenMaxTries; i++) {
const token = randomstring.generate(self.tokenLength) const token = randomstring.generate(self.tokenLength)
if (self.onHold.has(token)) if (self.onHold.has(token)) continue
continue
// Put token on-hold (wait for it to be inserted to DB) // Put token on-hold (wait for it to be inserted to DB)
self.onHold.add(token) self.onHold.add(token)
@ -40,8 +39,7 @@ self.verify = async (req, res, next) => {
? req.body.token.trim() ? req.body.token.trim()
: '' : ''
if (!token) if (!token) return res.json({ success: false, description: 'No token provided.' })
return res.json({ success: false, description: 'No token provided.' })
try { try {
const user = await db.table('users') const user = await db.table('users')
@ -49,8 +47,7 @@ self.verify = async (req, res, next) => {
.select('username', 'permission') .select('username', 'permission')
.first() .first()
if (!user) if (!user) return res.json({ success: false, description: 'Invalid token.' })
return res.json({ success: false, description: 'Invalid token.' })
const obj = { const obj = {
success: true, success: true,
@ -76,8 +73,12 @@ self.change = async (req, res, next) => {
if (!user) return if (!user) return
const newToken = await self.generateUniqueToken() const newToken = await self.generateUniqueToken()
if (!newToken) if (!newToken) {
return res.json({ success: false, description: 'Sorry, we could not allocate a unique token. Try again?' }) return res.json({
success: false,
description: 'Sorry, we could not allocate a unique token. Try again?'
})
}
try { try {
await db.table('users') await db.table('users')

File diff suppressed because it is too large Load Diff

View File

@ -18,7 +18,9 @@ const self = {
chunkSize: config.uploads.scan.chunkSize || 64 * 1024, chunkSize: config.uploads.scan.chunkSize || 64 * 1024,
groupBypass: config.uploads.scan.groupBypass || null, groupBypass: config.uploads.scan.groupBypass || null,
whitelistExtensions: (Array.isArray(config.uploads.scan.whitelistExtensions) && whitelistExtensions: (Array.isArray(config.uploads.scan.whitelistExtensions) &&
config.uploads.scan.whitelistExtensions.length) ? config.uploads.scan.whitelistExtensions : null, config.uploads.scan.whitelistExtensions.length)
? config.uploads.scan.whitelistExtensions
: null,
maxSize: (parseInt(config.uploads.scan.maxSize) * 1e6) || null maxSize: (parseInt(config.uploads.scan.maxSize) * 1e6) || null
}, },
gitHash: null, gitHash: null,
@ -92,14 +94,16 @@ self.extname = filename => {
} }
// check against extensions that must be preserved // check against extensions that must be preserved
for (const extPreserve of extPreserves) for (const extPreserve of extPreserves) {
if (lower.endsWith(extPreserve)) { if (lower.endsWith(extPreserve)) {
extname = extPreserve extname = extPreserve
break break
} }
}
if (!extname) if (!extname) {
extname = lower.slice(lower.lastIndexOf('.') - lower.length) // path.extname(lower) extname = lower.slice(lower.lastIndexOf('.') - lower.length) // path.extname(lower)
}
return extname + multi return extname + multi
} }
@ -110,14 +114,12 @@ self.escape = string => {
// Copyright(c) 2015 Andreas Lubbe // Copyright(c) 2015 Andreas Lubbe
// Copyright(c) 2015 Tiancheng "Timothy" Gu // Copyright(c) 2015 Tiancheng "Timothy" Gu
if (!string) if (!string) return string
return string
const str = String(string) const str = String(string)
const match = /["'&<>]/.exec(str) const match = /["'&<>]/.exec(str)
if (!match) if (!match) return str
return str
let escape let escape
let html = '' let html = ''
@ -145,8 +147,9 @@ self.escape = string => {
continue continue
} }
if (lastIndex !== index) if (lastIndex !== index) {
html += str.substring(lastIndex, index) html += str.substring(lastIndex, index)
}
lastIndex = index + 1 lastIndex = index + 1
html += escape html += escape
@ -203,16 +206,16 @@ self.generateThumbs = async (name, extname, force) => {
// Check if thumbnail already exists // Check if thumbnail already exists
try { try {
const lstat = await paths.lstat(thumbname) const lstat = await paths.lstat(thumbname)
if (lstat.isSymbolicLink()) if (lstat.isSymbolicLink()) {
// Unlink if symlink (should be symlink to the placeholder) // Unlink if symlink (should be symlink to the placeholder)
await paths.unlink(thumbname) await paths.unlink(thumbname)
else if (!force) } else if (!force) {
// Continue only if it does not exist, unless forced to // Continue only if it does not exist, unless forced to
return true return true
}
} catch (error) { } catch (error) {
// Re-throw error // Re-throw error
if (error.code !== 'ENOENT') if (error.code !== 'ENOENT') throw error
throw error
} }
// Full path to input file // Full path to input file
@ -257,12 +260,14 @@ self.generateThumbs = async (name, extname, force) => {
const metadata = await self.ffprobe(input) const metadata = await self.ffprobe(input)
const duration = parseInt(metadata.format.duration) const duration = parseInt(metadata.format.duration)
if (isNaN(duration)) if (isNaN(duration)) {
throw 'Warning: File does not have valid duration metadata' throw 'Warning: File does not have valid duration metadata'
}
const videoStream = metadata.streams && metadata.streams.find(s => s.codec_type === 'video') const videoStream = metadata.streams && metadata.streams.find(s => s.codec_type === 'video')
if (!videoStream || !videoStream.width || !videoStream.height) if (!videoStream || !videoStream.width || !videoStream.height) {
throw 'Warning: File does not have valid video stream metadata' throw 'Warning: File does not have valid video stream metadata'
}
await new Promise((resolve, reject) => { await new Promise((resolve, reject) => {
ffmpeg(input) ffmpeg(input)
@ -287,10 +292,11 @@ self.generateThumbs = async (name, extname, force) => {
await paths.lstat(thumbname) await paths.lstat(thumbname)
return true return true
} catch (err) { } catch (err) {
if (err.code === 'ENOENT') if (err.code === 'ENOENT') {
throw error || 'Warning: FFMPEG exited with empty output file' throw error || 'Warning: FFMPEG exited with empty output file'
else } else {
throw error || err throw error || err
}
} }
}) })
} else { } else {
@ -361,8 +367,7 @@ self.unlinkFile = async (filename, predb) => {
await paths.unlink(path.join(paths.uploads, filename)) await paths.unlink(path.join(paths.uploads, filename))
} catch (error) { } catch (error) {
// Return true if file does not exist // Return true if file does not exist
if (error.code !== 'ENOENT') if (error.code !== 'ENOENT') throw error
throw error
} }
const identifier = filename.split('.')[0] const identifier = filename.split('.')[0]
@ -375,26 +380,26 @@ self.unlinkFile = async (filename, predb) => {
} }
const extname = self.extname(filename) const extname = self.extname(filename)
if (self.imageExts.includes(extname) || self.videoExts.includes(extname)) if (self.imageExts.includes(extname) || self.videoExts.includes(extname)) {
try { try {
await paths.unlink(path.join(paths.thumbs, `${identifier}.png`)) await paths.unlink(path.join(paths.thumbs, `${identifier}.png`))
} catch (error) { } catch (error) {
if (error.code !== 'ENOENT') if (error.code !== 'ENOENT') throw error
throw error
} }
}
} }
self.bulkDeleteFromDb = async (field, values, user) => { self.bulkDeleteFromDb = async (field, values, user) => {
// Always return an empty array on failure // Always return an empty array on failure
if (!user || !['id', 'name'].includes(field) || !values.length) if (!user || !['id', 'name'].includes(field) || !values.length) return []
return []
// SQLITE_LIMIT_VARIABLE_NUMBER, which defaults to 999 // SQLITE_LIMIT_VARIABLE_NUMBER, which defaults to 999
// Read more: https://www.sqlite.org/limits.html // Read more: https://www.sqlite.org/limits.html
const MAX_VARIABLES_CHUNK_SIZE = 999 const MAX_VARIABLES_CHUNK_SIZE = 999
const chunks = [] const chunks = []
while (values.length) while (values.length) {
chunks.push(values.splice(0, MAX_VARIABLES_CHUNK_SIZE)) chunks.push(values.splice(0, MAX_VARIABLES_CHUNK_SIZE))
}
const failed = [] const failed = []
const ismoderator = perms.is(user, 'moderator') const ismoderator = perms.is(user, 'moderator')
@ -407,8 +412,9 @@ self.bulkDeleteFromDb = async (field, values, user) => {
const files = await db.table('files') const files = await db.table('files')
.whereIn(field, chunk) .whereIn(field, chunk)
.where(function () { .where(function () {
if (!ismoderator) if (!ismoderator) {
this.where('userid', user.id) this.where('userid', user.id)
}
}) })
// Push files that could not be found in db // Push files that could not be found in db
@ -435,17 +441,19 @@ self.bulkDeleteFromDb = async (field, values, user) => {
.del() .del()
self.invalidateStatsCache('uploads') self.invalidateStatsCache('uploads')
if (self.idSet) if (self.idSet) {
unlinked.forEach(file => { unlinked.forEach(file => {
const identifier = file.name.split('.')[0] const identifier = file.name.split('.')[0]
self.idSet.delete(identifier) self.idSet.delete(identifier)
// logger.log(`Removed ${identifier} from identifiers cache (bulkDeleteFromDb)`) // logger.log(`Removed ${identifier} from identifiers cache (bulkDeleteFromDb)`)
}) })
}
// Push album ids // Push album ids
unlinked.forEach(file => { unlinked.forEach(file => {
if (file.albumid && !albumids.includes(file.albumid)) if (file.albumid && !albumids.includes(file.albumid)) {
albumids.push(file.albumid) albumids.push(file.albumid)
}
}) })
// Push unlinked files // Push unlinked files
@ -463,13 +471,16 @@ self.bulkDeleteFromDb = async (field, values, user) => {
} }
// Purge Cloudflare's cache if necessary, but do not wait // Purge Cloudflare's cache if necessary, but do not wait
if (config.cloudflare.purgeCache) if (config.cloudflare.purgeCache) {
self.purgeCloudflareCache(unlinkeds.map(file => file.name), true, true) self.purgeCloudflareCache(unlinkeds.map(file => file.name), true, true)
.then(results => { .then(results => {
for (const result of results) for (const result of results) {
if (result.errors.length) if (result.errors.length) {
result.errors.forEach(error => logger.error(`[CF]: ${error}`)) result.errors.forEach(error => logger.error(`[CF]: ${error}`))
}
}
}) })
}
} }
} catch (error) { } catch (error) {
logger.error(error) logger.error(error)
@ -480,12 +491,15 @@ self.bulkDeleteFromDb = async (field, values, user) => {
self.purgeCloudflareCache = async (names, uploads, thumbs) => { self.purgeCloudflareCache = async (names, uploads, thumbs) => {
const errors = [] const errors = []
if (!cloudflareAuth) if (!cloudflareAuth) {
errors.push('Cloudflare auth is incomplete or missing') errors.push('Cloudflare auth is incomplete or missing')
if (!Array.isArray(names) || !names.length) }
if (!Array.isArray(names) || !names.length) {
errors.push('Names array is invalid or empty') errors.push('Names array is invalid or empty')
if (errors.length) }
if (errors.length) {
return [{ success: false, files: [], errors }] return [{ success: false, files: [], errors }]
}
let domain = config.domain let domain = config.domain
if (!uploads) domain = config.homeDomain if (!uploads) domain = config.homeDomain
@ -495,8 +509,9 @@ self.purgeCloudflareCache = async (names, uploads, thumbs) => {
if (uploads) { if (uploads) {
const url = `${domain}/${name}` const url = `${domain}/${name}`
const extname = self.extname(name) const extname = self.extname(name)
if (thumbs && self.mayGenerateThumb(extname)) if (thumbs && self.mayGenerateThumb(extname)) {
thumbNames.push(`${domain}/thumbs/${name.slice(0, -extname.length)}.png`) thumbNames.push(`${domain}/thumbs/${name.slice(0, -extname.length)}.png`)
}
return url return url
} else { } else {
return name === 'home' ? domain : `${domain}/${name}` return name === 'home' ? domain : `${domain}/${name}`
@ -509,8 +524,9 @@ self.purgeCloudflareCache = async (names, uploads, thumbs) => {
// TODO: Handle API rate limits // TODO: Handle API rate limits
const MAX_LENGTH = 30 const MAX_LENGTH = 30
const chunks = [] const chunks = []
while (names.length) while (names.length) {
chunks.push(names.splice(0, MAX_LENGTH)) chunks.push(names.splice(0, MAX_LENGTH))
}
const url = `https://api.cloudflare.com/client/v4/zones/${config.cloudflare.zoneId}/purge_cache` const url = `https://api.cloudflare.com/client/v4/zones/${config.cloudflare.zoneId}/purge_cache`
const results = [] const results = []
@ -543,8 +559,9 @@ self.purgeCloudflareCache = async (names, uploads, thumbs) => {
const response = await purge.json() const response = await purge.json()
result.success = response.success result.success = response.success
if (Array.isArray(response.errors) && response.errors.length) if (Array.isArray(response.errors) && response.errors.length) {
result.errors = response.errors.map(error => `${error.code}: ${error.message}`) result.errors = response.errors.map(error => `${error.code}: ${error.message}`)
}
} catch (error) { } catch (error) {
result.errors = [error.toString()] result.errors = [error.toString()]
} }
@ -642,7 +659,7 @@ self.stats = async (req, res, next) => {
} }
// Disk usage, only for Linux platform // Disk usage, only for Linux platform
if (os.platform === 'linux') if (os.platform === 'linux') {
if (!statsCache.disk.cache && statsCache.disk.generating) { if (!statsCache.disk.cache && statsCache.disk.generating) {
stats.disk = false stats.disk = false
} else if (((Date.now() - statsCache.disk.generatedAt) <= 60000) || statsCache.disk.generating) { } else if (((Date.now() - statsCache.disk.generatedAt) <= 60000) || statsCache.disk.generating) {
@ -727,8 +744,9 @@ self.stats = async (req, res, next) => {
stats.disk[basename] = parseInt(formatted[0]) stats.disk[basename] = parseInt(formatted[0])
// Add to types if necessary // Add to types if necessary
if (!stats.disk._types.byte.includes(basename)) if (!stats.disk._types.byte.includes(basename)) {
stats.disk._types.byte.push(basename) stats.disk._types.byte.push(basename)
}
}) })
const stderr = [] const stderr = []
@ -786,6 +804,7 @@ self.stats = async (req, res, next) => {
statsCache.disk.cache = stats.disk statsCache.disk.cache = stats.disk
statsCache.disk.generating = false statsCache.disk.generating = false
} }
}
// Uploads // Uploads
if (!statsCache.uploads.cache && statsCache.uploads.generating) { if (!statsCache.uploads.cache && statsCache.uploads.generating) {
@ -812,8 +831,9 @@ self.stats = async (req, res, next) => {
stats.uploads.total = uploads.length stats.uploads.total = uploads.length
stats.uploads.sizeInDb = uploads.reduce((acc, upload) => acc + parseInt(upload.size), 0) stats.uploads.sizeInDb = uploads.reduce((acc, upload) => acc + parseInt(upload.size), 0)
// Add type information for the new column // Add type information for the new column
if (!Array.isArray(stats.uploads._types.byte)) if (!Array.isArray(stats.uploads._types.byte)) {
stats.uploads._types.byte = [] stats.uploads._types.byte = []
}
stats.uploads._types.byte.push('sizeInDb') stats.uploads._types.byte.push('sizeInDb')
} else { } else {
stats.uploads.total = await db.table('files') stats.uploads.total = await db.table('files')
@ -823,16 +843,18 @@ self.stats = async (req, res, next) => {
stats.uploads.images = await db.table('files') stats.uploads.images = await db.table('files')
.where(function () { .where(function () {
for (const ext of self.imageExts) for (const ext of self.imageExts) {
this.orWhere('name', 'like', `%${ext}`) this.orWhere('name', 'like', `%${ext}`)
}
}) })
.count('id as count') .count('id as count')
.then(rows => rows[0].count) .then(rows => rows[0].count)
stats.uploads.videos = await db.table('files') stats.uploads.videos = await db.table('files')
.where(function () { .where(function () {
for (const ext of self.videoExts) for (const ext of self.videoExts) {
this.orWhere('name', 'like', `%${ext}`) this.orWhere('name', 'like', `%${ext}`)
}
}) })
.count('id as count') .count('id as count')
.then(rows => rows[0].count) .then(rows => rows[0].count)
@ -870,16 +892,18 @@ self.stats = async (req, res, next) => {
const users = await db.table('users') const users = await db.table('users')
stats.users.total = users.length stats.users.total = users.length
for (const user of users) { for (const user of users) {
if (user.enabled === false || user.enabled === 0) if (user.enabled === false || user.enabled === 0) {
stats.users.disabled++ stats.users.disabled++
}
// This may be inaccurate on installations with customized permissions // This may be inaccurate on installations with customized permissions
user.permission = user.permission || 0 user.permission = user.permission || 0
for (const p of permissionKeys) for (const p of permissionKeys) {
if (user.permission === perms.permissions[p]) { if (user.permission === perms.permissions[p]) {
stats.users[p]++ stats.users[p]++
break break
} }
}
} }
// Update cache // Update cache
@ -926,8 +950,7 @@ self.stats = async (req, res, next) => {
stats.albums.zipGenerated++ stats.albums.zipGenerated++
} catch (error) { } catch (error) {
// Re-throw error // Re-throw error
if (error.code !== 'ENOENT') if (error.code !== 'ENOENT') throw error
throw error
} }
})) }))

View File

@ -29,8 +29,9 @@ const postcssPlugins = [
sass.compiler = sassCompiler sass.compiler = sassCompiler
// Minify on production // Minify on production
if (process.env.NODE_ENV !== 'development') if (process.env.NODE_ENV !== 'development') {
postcssPlugins.push(cssnano()) postcssPlugins.push(cssnano())
}
/** TASKS: LINT */ /** TASKS: LINT */

View File

@ -54,8 +54,9 @@ self.debug = (...args) => {
Object.assign(options, args[args.length - 1]) Object.assign(options, args[args.length - 1])
args.splice(args.length - 1, 1) args.splice(args.length - 1, 1)
} }
for (const arg of args) for (const arg of args) {
console.log(inspect(arg, options)) console.log(inspect(arg, options))
}
} }
module.exports = self module.exports = self

View File

@ -36,11 +36,13 @@ safe.use(helmet({
hsts: false hsts: false
})) }))
if (config.hsts instanceof Object && Object.keys(config.hsts).length) if (config.hsts instanceof Object && Object.keys(config.hsts).length) {
safe.use(helmet.hsts(config.hsts)) safe.use(helmet.hsts(config.hsts))
}
if (config.trustProxy) if (config.trustProxy) {
safe.set('trust proxy', 1) safe.set('trust proxy', 1)
}
// https://mozilla.github.io/nunjucks/api.html#configure // https://mozilla.github.io/nunjucks/api.html#configure
nunjucks.configure('views', { nunjucks.configure('views', {
@ -52,12 +54,14 @@ safe.set('view engine', 'njk')
safe.enable('view cache') safe.enable('view cache')
// Configure rate limits // Configure rate limits
if (Array.isArray(config.rateLimits) && config.rateLimits.length) if (Array.isArray(config.rateLimits) && config.rateLimits.length) {
for (const rateLimit of config.rateLimits) { for (const rateLimit of config.rateLimits) {
const limiter = new RateLimit(rateLimit.config) const limiter = new RateLimit(rateLimit.config)
for (const route of rateLimit.routes) for (const route of rateLimit.routes) {
safe.use(route, limiter) safe.use(route, limiter)
}
} }
}
safe.use(bodyParser.urlencoded({ extended: true })) safe.use(bodyParser.urlencoded({ extended: true }))
safe.use(bodyParser.json()) safe.use(bodyParser.json())
@ -117,24 +121,27 @@ if (config.cacheControl) {
// If using CDN, cache public pages in CDN // If using CDN, cache public pages in CDN
if (config.cacheControl !== 2) { if (config.cacheControl !== 2) {
cdnPages.push('api/check') cdnPages.push('api/check')
for (const page of cdnPages) for (const page of cdnPages) {
safe.use(`/${page === 'home' ? '' : page}`, (req, res, next) => { safe.use(`/${page === 'home' ? '' : page}`, (req, res, next) => {
res.set('Cache-Control', cacheControls.cdn) res.set('Cache-Control', cacheControls.cdn)
next() next()
}) })
}
} }
// If serving uploads with node // If serving uploads with node
if (config.serveFilesWithNode) if (config.serveFilesWithNode) {
initServeStaticUploads({ initServeStaticUploads({
setHeaders: res => { setHeaders: res => {
res.set('Access-Control-Allow-Origin', '*') res.set('Access-Control-Allow-Origin', '*')
// If using CDN, cache uploads in CDN as well // If using CDN, cache uploads in CDN as well
// Use with cloudflare.purgeCache enabled in config file // Use with cloudflare.purgeCache enabled in config file
if (config.cacheControl !== 2) if (config.cacheControl !== 2) {
res.set('Cache-Control', cacheControls.cdn) res.set('Cache-Control', cacheControls.cdn)
}
} }
}) })
}
// Function for static assets. // Function for static assets.
// This requires the assets to use version in their query string, // This requires the assets to use version in their query string,
@ -148,10 +155,11 @@ if (config.cacheControl) {
safe.use(['/api/album/zip'], (req, res, next) => { safe.use(['/api/album/zip'], (req, res, next) => {
res.set('Access-Control-Allow-Origin', '*') res.set('Access-Control-Allow-Origin', '*')
const versionString = parseInt(req.query.v) const versionString = parseInt(req.query.v)
if (versionString > 0) if (versionString > 0) {
res.set('Cache-Control', cacheControls.static) res.set('Cache-Control', cacheControls.static)
else } else {
res.set('Cache-Control', cacheControls.disable) res.set('Cache-Control', cacheControls.disable)
}
next() next()
}) })
} else if (config.serveFilesWithNode) { } else if (config.serveFilesWithNode) {
@ -182,32 +190,36 @@ safe.use('/api', api)
// Re-map version strings if cache control is enabled (safe.fiery.me) // Re-map version strings if cache control is enabled (safe.fiery.me)
utils.versionStrings = {} utils.versionStrings = {}
if (config.cacheControl) { if (config.cacheControl) {
for (const type in versions) for (const type in versions) {
utils.versionStrings[type] = `?_=${versions[type]}` utils.versionStrings[type] = `?_=${versions[type]}`
if (versions['1']) }
if (versions['1']) {
utils.clientVersion = versions['1'] utils.clientVersion = versions['1']
}
} }
// Cookie Policy // Cookie Policy
if (config.cookiePolicy) if (config.cookiePolicy) {
config.pages.push('cookiepolicy') config.pages.push('cookiepolicy')
}
// Check for custom pages, otherwise fallback to Nunjucks templates // Check for custom pages, otherwise fallback to Nunjucks templates
for (const page of config.pages) { for (const page of config.pages) {
const customPage = path.join(paths.customPages, `${page}.html`) const customPage = path.join(paths.customPages, `${page}.html`)
if (!await paths.access(customPage).catch(() => true)) if (!await paths.access(customPage).catch(() => true)) {
safe.get(`/${page === 'home' ? '' : page}`, (req, res, next) => res.sendFile(customPage)) safe.get(`/${page === 'home' ? '' : page}`, (req, res, next) => res.sendFile(customPage))
else if (page === 'home') } else if (page === 'home') {
safe.get('/', (req, res, next) => res.render(page, { safe.get('/', (req, res, next) => res.render(page, {
config, config,
versions: utils.versionStrings, versions: utils.versionStrings,
gitHash: utils.gitHash gitHash: utils.gitHash
})) }))
else } else {
safe.get(`/${page}`, (req, res, next) => res.render(page, { safe.get(`/${page}`, (req, res, next) => res.render(page, {
config, config,
versions: utils.versionStrings versions: utils.versionStrings
})) }))
}
} }
// Error pages // Error pages
@ -240,8 +252,9 @@ safe.use('/api', api)
logger.log(`${ip}:${port} ${version}`) logger.log(`${ip}:${port} ${version}`)
utils.clamd.scanner = clamd.createScanner(ip, port) utils.clamd.scanner = clamd.createScanner(ip, port)
if (!utils.clamd.scanner) if (!utils.clamd.scanner) {
throw 'Could not create clamd scanner' throw 'Could not create clamd scanner'
}
} }
// Cache file identifiers // Cache file identifiers
@ -260,7 +273,7 @@ safe.use('/api', api)
// Cache control (safe.fiery.me) // Cache control (safe.fiery.me)
// Purge Cloudflare cache // Purge Cloudflare cache
if (config.cacheControl && config.cacheControl !== 2) if (config.cacheControl && config.cacheControl !== 2) {
if (config.cloudflare.purgeCache) { if (config.cloudflare.purgeCache) {
logger.log('Cache control enabled, purging Cloudflare\'s cache...') logger.log('Cache control enabled, purging Cloudflare\'s cache...')
const results = await utils.purgeCloudflareCache(cdnPages) const results = await utils.purgeCloudflareCache(cdnPages)
@ -274,11 +287,13 @@ safe.use('/api', api)
} }
succeeded += result.files.length succeeded += result.files.length
} }
if (!errored) if (!errored) {
logger.log(`Successfully purged ${succeeded} cache`) logger.log(`Successfully purged ${succeeded} cache`)
}
} else { } else {
logger.log('Cache control enabled without Cloudflare\'s cache purging') logger.log('Cache control enabled without Cloudflare\'s cache purging')
} }
}
// Temporary uploads (only check for expired uploads if config.uploads.temporaryUploadsInterval is also set) // Temporary uploads (only check for expired uploads if config.uploads.temporaryUploadsInterval is also set)
if (Array.isArray(config.uploads.temporaryUploadAges) && if (Array.isArray(config.uploads.temporaryUploadAges) &&
@ -286,8 +301,7 @@ safe.use('/api', api)
config.uploads.temporaryUploadsInterval) { config.uploads.temporaryUploadsInterval) {
let temporaryUploadsInProgress = false let temporaryUploadsInProgress = false
const temporaryUploadCheck = async () => { const temporaryUploadCheck = async () => {
if (temporaryUploadsInProgress) if (temporaryUploadsInProgress) return
return
temporaryUploadsInProgress = true temporaryUploadsInProgress = true
try { try {
@ -295,8 +309,9 @@ safe.use('/api', api)
if (result.expired.length) { if (result.expired.length) {
let logMessage = `Expired uploads: ${result.expired.length} deleted` let logMessage = `Expired uploads: ${result.expired.length} deleted`
if (result.failed.length) if (result.failed.length) {
logMessage += `, ${result.failed.length} errored` logMessage += `, ${result.failed.length} errored`
}
logger.log(logMessage) logger.log(logMessage)
} }
@ -321,10 +336,8 @@ safe.use('/api', api)
prompt: '' prompt: ''
}).on('line', line => { }).on('line', line => {
try { try {
if (line === 'rs') if (line === 'rs') return
return if (line === '.exit') return process.exit(0)
if (line === '.exit')
return process.exit(0)
// eslint-disable-next-line no-eval // eslint-disable-next-line no-eval
logger.log(eval(line)) logger.log(eval(line))
} catch (error) { } catch (error) {

View File

@ -12,11 +12,6 @@ module.exports = {
'plugin:compat/recommended' 'plugin:compat/recommended'
], ],
rules: { rules: {
curly: [
'error',
'multi',
'consistent'
],
'object-shorthand': [ 'object-shorthand': [
'error', 'error',
'always' 'always'

View File

@ -9,8 +9,9 @@ const page = {
window.addEventListener('DOMContentLoaded', () => { window.addEventListener('DOMContentLoaded', () => {
const elements = document.querySelectorAll('.file-size') const elements = document.querySelectorAll('.file-size')
for (let i = 0; i < elements.length; i++) for (let i = 0; i < elements.length; i++) {
elements[i].innerHTML = page.getPrettyBytes(parseInt(elements[i].innerHTML.replace(/\s*B$/i, ''))) elements[i].innerHTML = page.getPrettyBytes(parseInt(elements[i].innerHTML.replace(/\s*B$/i, '')))
}
page.lazyLoad = new LazyLoad() page.lazyLoad = new LazyLoad()
}) })

View File

@ -15,16 +15,13 @@ const page = {
page.unhide = () => { page.unhide = () => {
const loaderSection = document.querySelector('#loader') const loaderSection = document.querySelector('#loader')
if (loaderSection) if (loaderSection) loaderSection.classList.add('is-hidden')
loaderSection.classList.add('is-hidden')
const loginSection = document.querySelector('#login.is-hidden') const loginSection = document.querySelector('#login.is-hidden')
if (loginSection) if (loginSection) loginSection.classList.remove('is-hidden')
loginSection.classList.remove('is-hidden')
const floatingBtn = document.querySelector('.floating-home-button.is-hidden') const floatingBtn = document.querySelector('.floating-home-button.is-hidden')
if (floatingBtn) if (floatingBtn) floatingBtn.classList.remove('is-hidden')
floatingBtn.classList.remove('is-hidden')
} }
// Handler for Axios errors // Handler for Axios errors
@ -54,12 +51,10 @@ page.onAxiosError = error => {
page.do = (dest, trigger) => { page.do = (dest, trigger) => {
const user = page.user.value.trim() const user = page.user.value.trim()
if (!user) if (!user) return swal('An error occurred!', 'You need to specify a username.', 'error')
return swal('An error occurred!', 'You need to specify a username.', 'error')
const pass = page.pass.value.trim() const pass = page.pass.value.trim()
if (!pass) if (!pass) return swal('An error occurred!', 'You need to specify a password.', 'error')
return swal('An error occurred!', 'You need to specify a password.', 'error')
trigger.classList.add('is-loading') trigger.classList.add('is-loading')
axios.post(`api/${dest}`, { axios.post(`api/${dest}`, {
@ -107,23 +102,24 @@ window.addEventListener('DOMContentLoaded', () => {
}) })
const loginBtn = document.querySelector('#loginBtn') const loginBtn = document.querySelector('#loginBtn')
if (loginBtn) if (loginBtn) {
loginBtn.addEventListener('click', event => { loginBtn.addEventListener('click', event => {
if (!form.checkValidity()) return if (!form.checkValidity()) return
page.do('login', event.currentTarget) page.do('login', event.currentTarget)
}) })
}
const registerBtn = document.querySelector('#registerBtn') const registerBtn = document.querySelector('#registerBtn')
if (registerBtn) if (registerBtn) {
registerBtn.addEventListener('click', event => { registerBtn.addEventListener('click', event => {
if (!form.checkValidity()) if (!form.checkValidity()) {
// Workaround for browsers to display native form error messages // Workaround for browsers to display native form error messages
return loginBtn.click() return loginBtn.click()
}
page.do('register', event.currentTarget) page.do('register', event.currentTarget)
}) })
}
if (page.token) if (page.token) page.verify()
page.verify() else page.unhide()
else
page.unhide()
}) })

File diff suppressed because it is too large Load Diff

View File

@ -81,10 +81,8 @@ page.onInitError = error => {
window.location.reload() window.location.reload()
}) })
if (error.response) if (error.response) page.onAxiosError(error)
page.onAxiosError(error) else page.onError(error)
else
page.onError(error)
} }
// Handler for regular JS errors // Handler for regular JS errors
@ -102,8 +100,7 @@ page.onError = error => {
// Handler for Axios errors // Handler for Axios errors
page.onAxiosError = (error, cont) => { page.onAxiosError = (error, cont) => {
if (!cont) if (!cont) console.error(error)
console.error(error)
// Better Cloudflare errors // Better Cloudflare errors
const cloudflareErrors = { const cloudflareErrors = {
@ -138,7 +135,7 @@ page.onAxiosError = (error, cont) => {
page.checkClientVersion = apiVersion => { page.checkClientVersion = apiVersion => {
const self = document.querySelector('#mainScript') const self = document.querySelector('#mainScript')
const match = self.src.match(/\?_=(\d+)$/) const match = self.src.match(/\?_=(\d+)$/)
if (match && match[1] && match[1] !== apiVersion) if (match && match[1] && match[1] !== apiVersion) {
return swal({ return swal({
title: 'Update detected!', title: 'Update detected!',
text: 'Client assets have been updated. Reload to display the latest version?', text: 'Client assets have been updated. Reload to display the latest version?',
@ -152,27 +149,23 @@ page.checkClientVersion = apiVersion => {
}).then(() => { }).then(() => {
window.location.reload() window.location.reload()
}) })
}
} }
page.checkIfPublic = () => { page.checkIfPublic = () => {
return axios.get('api/check', { return axios.get('api/check', {
onDownloadProgress: () => { onDownloadProgress: () => {
// Only do render and/or newsfeed after this request has been initiated to avoid blocking // Only do render and/or newsfeed after this request has been initiated to avoid blocking
/* global render */ /* global render */
if (typeof render !== 'undefined' && !render.done) if (typeof render !== 'undefined' && !render.done) render.do()
render.do()
/* global newsfeed */ /* global newsfeed */
if (typeof newsfeed !== 'undefined' && !newsfeed.done) if (typeof newsfeed !== 'undefined' && !newsfeed.done) newsfeed.do()
newsfeed.do() if (!page.apiChecked) page.apiChecked = true
if (!page.apiChecked)
page.apiChecked = true
} }
}).then(response => { }).then(response => {
if (response.data.version) if (response.data.version) {
page.checkClientVersion(response.data.version) page.checkClientVersion(response.data.version)
}
page.private = response.data.private page.private = response.data.private
page.enableUserAccounts = response.data.enableUserAccounts page.enableUserAccounts = response.data.enableUserAccounts
@ -193,25 +186,27 @@ page.checkIfPublic = () => {
} }
page.preparePage = () => { page.preparePage = () => {
if (page.private) if (page.private) {
if (page.token) { if (page.token) {
return page.verifyToken(page.token, true) return page.verifyToken(page.token, true)
} else { } else {
const button = document.querySelector('#loginToUpload') const button = document.querySelector('#loginToUpload')
button.href = 'auth' button.href = 'auth'
button.classList.remove('is-loading') button.classList.remove('is-loading')
if (page.enableUserAccounts) if (page.enableUserAccounts) {
button.innerText = 'Anonymous upload is disabled.\nLog in or register to upload.' button.innerText = 'Anonymous upload is disabled.\nLog in or register to upload.'
else } else {
button.innerText = 'Running in private mode.\nLog in to upload.' button.innerText = 'Running in private mode.\nLog in to upload.'
}
} }
else } else {
return page.prepareUpload() return page.prepareUpload()
}
} }
page.verifyToken = (token, reloadOnError) => { page.verifyToken = (token, reloadOnError) => {
return axios.post('api/tokens/verify', { token }).then(response => { return axios.post('api/tokens/verify', { token }).then(response => {
if (response.data.success === false) if (response.data.success === false) {
return swal({ return swal({
title: 'An error occurred!', title: 'An error occurred!',
text: response.data.description, text: response.data.description,
@ -221,6 +216,7 @@ page.verifyToken = (token, reloadOnError) => {
localStorage.removeItem('token') localStorage.removeItem('token')
window.location.reload() window.location.reload()
}) })
}
localStorage[lsKeys.token] = token localStorage[lsKeys.token] = token
page.token = token page.token = token
@ -233,8 +229,7 @@ page.prepareUpload = () => {
if (page.token) { if (page.token) {
// Change /auth link to /dashboard // Change /auth link to /dashboard
const authLink = document.querySelector('#linksColumn a[href="auth"]') const authLink = document.querySelector('#linksColumn a[href="auth"]')
if (authLink) if (authLink) authLink.setAttribute('href', 'dashboard')
authLink.setAttribute('href', 'dashboard')
// Display the album selection // Display the album selection
document.querySelector('#albumDiv').classList.remove('is-hidden') document.querySelector('#albumDiv').classList.remove('is-hidden')
@ -243,8 +238,7 @@ page.prepareUpload = () => {
page.albumSelectOnChange = () => { page.albumSelectOnChange = () => {
page.album = parseInt(page.albumSelect.value) page.album = parseInt(page.albumSelect.value)
// Re-generate ShareX config file // Re-generate ShareX config file
if (typeof page.prepareShareX === 'function') if (typeof page.prepareShareX === 'function') page.prepareShareX()
page.prepareShareX()
} }
page.albumSelect.addEventListener('change', page.albumSelectOnChange) page.albumSelect.addEventListener('change', page.albumSelectOnChange)
@ -265,8 +259,7 @@ page.prepareUpload = () => {
page.prepareDropzone() page.prepareDropzone()
// Generate ShareX config file // Generate ShareX config file
if (typeof page.prepareShareX === 'function') if (typeof page.prepareShareX === 'function') page.prepareShareX()
page.prepareShareX()
// Prepare urls upload tab // Prepare urls upload tab
const urlMaxSize = document.querySelector('#urlMaxSize') const urlMaxSize = document.querySelector('#urlMaxSize')
@ -301,7 +294,7 @@ page.prepareUpload = () => {
} }
page.setActiveTab = index => { page.setActiveTab = index => {
for (let i = 0; i < page.tabs.length; i++) for (let i = 0; i < page.tabs.length; i++) {
if (i === index) { if (i === index) {
page.tabs[i].tab.classList.add('is-active') page.tabs[i].tab.classList.add('is-active')
page.tabs[i].content.classList.remove('is-hidden') page.tabs[i].content.classList.remove('is-hidden')
@ -310,15 +303,17 @@ page.setActiveTab = index => {
page.tabs[i].tab.classList.remove('is-active') page.tabs[i].tab.classList.remove('is-active')
page.tabs[i].content.classList.add('is-hidden') page.tabs[i].content.classList.add('is-hidden')
} }
}
} }
page.fetchAlbums = () => { page.fetchAlbums = () => {
return axios.get('api/albums', { headers: { token: page.token } }).then(response => { return axios.get('api/albums', { headers: { token: page.token } }).then(response => {
if (response.data.success === false) if (response.data.success === false) {
return swal('An error occurred!', response.data.description, 'error') return swal('An error occurred!', response.data.description, 'error')
}
// Create an option for each album // Create an option for each album
if (Array.isArray(response.data.albums) && response.data.albums.length) if (Array.isArray(response.data.albums) && response.data.albums.length) {
for (let i = 0; i < response.data.albums.length; i++) { for (let i = 0; i < response.data.albums.length; i++) {
const album = response.data.albums[i] const album = response.data.albums[i]
const option = document.createElement('option') const option = document.createElement('option')
@ -326,6 +321,7 @@ page.fetchAlbums = () => {
option.innerHTML = album.name option.innerHTML = album.name
page.albumSelect.appendChild(option) page.albumSelect.appendChild(option)
} }
}
}).catch(page.onInitError) }).catch(page.onInitError)
} }
@ -371,8 +367,7 @@ page.prepareDropzone = () => {
init () { init () {
this.on('addedfile', file => { this.on('addedfile', file => {
// Set active tab to file uploads, if necessary // Set active tab to file uploads, if necessary
if (page.activeTab !== 0) if (page.activeTab !== 0) page.setActiveTab(0)
page.setActiveTab(0)
// Add file entry // Add file entry
tabDiv.querySelector('.uploads').classList.remove('is-hidden') tabDiv.querySelector('.uploads').classList.remove('is-hidden')
@ -383,19 +378,21 @@ page.prepareDropzone = () => {
this.on('sending', (file, xhr) => { this.on('sending', (file, xhr) => {
// Add timeout listener (hacky method due to lack of built-in timeout handler) // Add timeout listener (hacky method due to lack of built-in timeout handler)
if (!xhr.ontimeout) if (!xhr.ontimeout) {
xhr.ontimeout = () => { xhr.ontimeout = () => {
const instances = page.dropzone.getUploadingFiles() const instances = page.dropzone.getUploadingFiles()
.filter(instance => instance.xhr === xhr) .filter(instance => instance.xhr === xhr)
page.dropzone._handleUploadError(instances, xhr, 'Connection timed out. Try to reduce upload chunk size.') page.dropzone._handleUploadError(instances, xhr, 'Connection timed out. Try to reduce upload chunk size.')
} }
}
// Attach necessary data for initial upload speed calculation // Attach necessary data for initial upload speed calculation
if (xhr._uplSpeedCalc === undefined) if (xhr._uplSpeedCalc === undefined) {
xhr._uplSpeedCalc = { xhr._uplSpeedCalc = {
lastSent: 0, lastSent: 0,
data: [{ timestamp: Date.now(), bytes: 0 }] data: [{ timestamp: Date.now(), bytes: 0 }]
} }
}
// If not chunked uploads, add extra headers // If not chunked uploads, add extra headers
if (!file.upload.chunked) { if (!file.upload.chunked) {
@ -405,10 +402,11 @@ page.prepareDropzone = () => {
if (page.stripTags !== null) xhr.setRequestHeader('striptags', page.stripTags) if (page.stripTags !== null) xhr.setRequestHeader('striptags', page.stripTags)
} }
if (!file.upload.chunked) if (!file.upload.chunked) {
file.previewElement.querySelector('.descriptive-progress').innerHTML = 'Uploading\u2026' file.previewElement.querySelector('.descriptive-progress').innerHTML = 'Uploading\u2026'
else if (file.upload.chunks.length === 1) } else if (file.upload.chunks.length === 1) {
file.previewElement.querySelector('.descriptive-progress').innerHTML = `Uploading chunk 1/${file.upload.totalChunkCount}\u2026` file.previewElement.querySelector('.descriptive-progress').innerHTML = `Uploading chunk 1/${file.upload.totalChunkCount}\u2026`
}
}) })
// Update descriptive progress // Update descriptive progress
@ -474,8 +472,7 @@ page.prepareDropzone = () => {
} }
// If not enough data // If not enough data
if (!fullSec) if (!fullSec) bytesPerSec = 1000 / elapsed * bytesPerSec
bytesPerSec = 1000 / elapsed * bytesPerSec
// Get pretty bytes // Get pretty bytes
prettyBytesPerSec = page.getPrettyBytes(bytesPerSec) prettyBytesPerSec = page.getPrettyBytes(bytesPerSec)
@ -495,15 +492,16 @@ page.prepareDropzone = () => {
file.previewElement.querySelector('.error').classList.remove('is-hidden') file.previewElement.querySelector('.error').classList.remove('is-hidden')
} }
if (Array.isArray(data.files) && data.files[0]) if (Array.isArray(data.files) && data.files[0]) {
page.updateTemplate(file, data.files[0]) page.updateTemplate(file, data.files[0])
}
}) })
this.on('error', (file, error, xhr) => { this.on('error', (file, error, xhr) => {
let err = error let err = error
if (typeof error === 'object' && error.description) if (typeof error === 'object' && error.description) {
err = error.description err = error.description
else if (xhr) } else if (xhr) {
// Formatting the Object is necessary since the function expect Axios errors // Formatting the Object is necessary since the function expect Axios errors
err = page.onAxiosError({ err = page.onAxiosError({
response: { response: {
@ -511,12 +509,14 @@ page.prepareDropzone = () => {
statusText: xhr.statusText statusText: xhr.statusText
} }
}, true).data.description }, true).data.description
else if (error instanceof Error) } else if (error instanceof Error) {
err = error.toString() err = error.toString()
}
// Clean up file size errors // Clean up file size errors
if (/^File is too big/.test(err) && /File too large/.test(err)) if (/^File is too big/.test(err) && /File too large/.test(err)) {
err = `File too large (${page.getPrettyBytes(file.size)}).` err = `File too large (${page.getPrettyBytes(file.size)}).`
}
page.updateTemplateIcon(file.previewElement, 'icon-block') page.updateTemplateIcon(file.previewElement, 'icon-block')
@ -556,8 +556,9 @@ page.prepareDropzone = () => {
file.previewElement.querySelector('.error').classList.remove('is-hidden') file.previewElement.querySelector('.error').classList.remove('is-hidden')
} }
if (response.data.files && response.data.files[0]) if (response.data.files && response.data.files[0]) {
page.updateTemplate(file, response.data.files[0]) page.updateTemplate(file, response.data.files[0])
}
return done() return done()
}) })
@ -572,8 +573,9 @@ page.addUrlsToQueue = () => {
return url.trim().length return url.trim().length
}) })
if (!urls.length) if (!urls.length) {
return swal('An error occurred!', 'You have not entered any URLs.', 'error') return swal('An error occurred!', 'You have not entered any URLs.', 'error')
}
const tabDiv = document.querySelector('#tab-urls') const tabDiv = document.querySelector('#tab-urls')
tabDiv.querySelector('.uploads').classList.remove('is-hidden') tabDiv.querySelector('.uploads').classList.remove('is-hidden')
@ -607,15 +609,17 @@ page.processUrlsQueue = () => {
if (data.success === false) { if (data.success === false) {
const match = data.description.match(/ over limit: (\d+)$/) const match = data.description.match(/ over limit: (\d+)$/)
if (match && match[1]) if (match && match[1]) {
data.description = `File exceeded limit of ${page.getPrettyBytes(match[1])}.` data.description = `File exceeded limit of ${page.getPrettyBytes(match[1])}.`
}
file.previewElement.querySelector('.error').innerHTML = data.description file.previewElement.querySelector('.error').innerHTML = data.description
file.previewElement.querySelector('.error').classList.remove('is-hidden') file.previewElement.querySelector('.error').classList.remove('is-hidden')
} }
if (Array.isArray(data.files) && data.files[0]) if (Array.isArray(data.files) && data.files[0]) {
page.updateTemplate(file, data.files[0]) page.updateTemplate(file, data.files[0])
}
page.activeUrlsQueue-- page.activeUrlsQueue--
return shiftQueue() return shiftQueue()
@ -673,7 +677,7 @@ page.updateTemplate = (file, response) => {
? exec[0].toLowerCase() ? exec[0].toLowerCase()
: null : null
if (page.imageExts.includes(extname)) if (page.imageExts.includes(extname)) {
if (page.previewImages) { if (page.previewImages) {
const img = file.previewElement.querySelector('img') const img = file.previewElement.querySelector('img')
img.setAttribute('alt', response.name || '') img.setAttribute('alt', response.name || '')
@ -689,10 +693,11 @@ page.updateTemplate = (file, response) => {
} else { } else {
page.updateTemplateIcon(file.previewElement, 'icon-picture') page.updateTemplateIcon(file.previewElement, 'icon-picture')
} }
else if (page.videoExts.includes(extname)) } else if (page.videoExts.includes(extname)) {
page.updateTemplateIcon(file.previewElement, 'icon-video') page.updateTemplateIcon(file.previewElement, 'icon-video')
else } else {
page.updateTemplateIcon(file.previewElement, 'icon-doc-inv') page.updateTemplateIcon(file.previewElement, 'icon-doc-inv')
}
if (response.expirydate) { if (response.expirydate) {
const expiryDate = file.previewElement.querySelector('.expiry-date') const expiryDate = file.previewElement.querySelector('.expiry-date')
@ -758,8 +763,9 @@ page.createAlbum = () => {
token: page.token token: page.token
} }
}).then(response => { }).then(response => {
if (response.data.success === false) if (response.data.success === false) {
return swal('An error occurred!', response.data.description, 'error') return swal('An error occurred!', response.data.description, 'error')
}
const option = document.createElement('option') const option = document.createElement('option')
page.albumSelect.appendChild(option) page.albumSelect.appendChild(option)
@ -865,8 +871,9 @@ page.prepareUploadConfig = () => {
valueHandler (value) { valueHandler (value) {
if (value === '0') { if (value === '0') {
const uploadFields = document.querySelectorAll('.tab-content > .uploads') const uploadFields = document.querySelectorAll('.tab-content > .uploads')
for (let i = 0; i < uploadFields.length; i++) for (let i = 0; i < uploadFields.length; i++) {
uploadFields[i].classList.add('is-reversed') uploadFields[i].classList.add('is-reversed')
}
} }
} }
}, },
@ -891,8 +898,9 @@ page.prepareUploadConfig = () => {
value: i === 0 ? 'default' : String(age), value: i === 0 ? 'default' : String(age),
text: page.getPrettyUploadAge(age) text: page.getPrettyUploadAge(age)
}) })
if (age === stored) if (age === stored) {
config.uploadAge.value = stored config.uploadAge.value = stored
}
} }
} }
@ -901,8 +909,9 @@ page.prepareUploadConfig = () => {
if (!page.fileIdentifierLength.force && if (!page.fileIdentifierLength.force &&
!isNaN(stored) && !isNaN(stored) &&
stored >= page.fileIdentifierLength.min && stored >= page.fileIdentifierLength.min &&
stored <= page.fileIdentifierLength.max) stored <= page.fileIdentifierLength.max) {
config.fileLength.value = stored config.fileLength.value = stored
}
} }
const tabContent = document.querySelector('#tab-config') const tabContent = document.querySelector('#tab-config')
@ -915,8 +924,7 @@ page.prepareUploadConfig = () => {
const conf = config[key] const conf = config[key]
// Skip only if display attribute is explicitly set to false // Skip only if display attribute is explicitly set to false
if (conf.display === false) if (conf.display === false) continue
continue
const field = document.createElement('div') const field = document.createElement('div')
field.className = 'field' field.className = 'field'
@ -927,24 +935,29 @@ page.prepareUploadConfig = () => {
value = conf.value value = conf.value
} else if (conf.number !== undefined) { } else if (conf.number !== undefined) {
const parsed = parseInt(localStorage[lsKeys[key]]) const parsed = parseInt(localStorage[lsKeys[key]])
if (!isNaN(parsed) && parsed <= conf.number.max && parsed >= conf.number.min) if (!isNaN(parsed) && parsed <= conf.number.max && parsed >= conf.number.min) {
value = parsed value = parsed
}
} else { } else {
const stored = localStorage[lsKeys[key]] const stored = localStorage[lsKeys[key]]
if (Array.isArray(conf.select)) if (Array.isArray(conf.select)) {
value = conf.select.find(sel => sel.value === stored) ? stored : undefined value = conf.select.find(sel => sel.value === stored)
else ? stored
: undefined
} else {
value = stored value = stored
}
} }
// If valueHandler function exists, defer to the function, // If valueHandler function exists, defer to the function,
// otherwise pass value to global page object // otherwise pass value to global page object
if (typeof conf.valueHandler === 'function') if (typeof conf.valueHandler === 'function') {
conf.valueHandler(value) conf.valueHandler(value)
else if (value !== undefined) } else if (value !== undefined) {
page[key] = value page[key] = value
else if (fallback[key] !== undefined) } else if (fallback[key] !== undefined) {
page[key] = fallback[key] page[key] = fallback[key]
}
} }
let control let control
@ -975,34 +988,34 @@ page.prepareUploadConfig = () => {
control.className = 'input is-fullwidth' control.className = 'input is-fullwidth'
control.type = 'number' control.type = 'number'
if (conf.number.min !== undefined) if (conf.number.min !== undefined) control.min = conf.number.min
control.min = conf.number.min if (conf.number.max !== undefined) control.max = conf.number.max
if (conf.number.max !== undefined) if (typeof value === 'number') control.value = value
control.max = conf.number.max else if (conf.number.default !== undefined) control.value = conf.number.default
if (typeof value === 'number')
control.value = value
else if (conf.number.default !== undefined)
control.value = conf.number.default
} }
let help let help
if (conf.disabled) { if (conf.disabled) {
if (Array.isArray(conf.select)) if (Array.isArray(conf.select)) {
control.querySelector('select').disabled = conf.disabled control.querySelector('select').disabled = conf.disabled
else } else {
control.disabled = conf.disabled control.disabled = conf.disabled
}
help = 'This option is currently not configurable.' help = 'This option is currently not configurable.'
} else if (typeof conf.help === 'string') { } else if (typeof conf.help === 'string') {
help = conf.help help = conf.help
} else if (conf.help === true && conf.number !== undefined) { } else if (conf.help === true && conf.number !== undefined) {
const tmp = [] const tmp = []
if (conf.number.default !== undefined) if (conf.number.default !== undefined) {
tmp.push(`Default is ${conf.number.default}${conf.number.suffix || ''}.`) tmp.push(`Default is ${conf.number.default}${conf.number.suffix || ''}.`)
if (conf.number.min !== undefined) }
if (conf.number.min !== undefined) {
tmp.push(`Min is ${conf.number.min}${conf.number.suffix || ''}.`) tmp.push(`Min is ${conf.number.min}${conf.number.suffix || ''}.`)
if (conf.number.max !== undefined) }
if (conf.number.max !== undefined) {
tmp.push(`Max is ${conf.number.max}${conf.number.suffix || ''}.`) tmp.push(`Max is ${conf.number.max}${conf.number.suffix || ''}.`)
}
help = tmp.join(' ') help = tmp.join(' ')
} }
@ -1036,8 +1049,7 @@ page.prepareUploadConfig = () => {
form.appendChild(submit) form.appendChild(submit)
form.querySelector('#saveConfig').addEventListener('click', () => { form.querySelector('#saveConfig').addEventListener('click', () => {
if (!form.checkValidity()) if (!form.checkValidity()) return
return
const keys = Object.keys(config) const keys = Object.keys(config)
.filter(key => config[key].display !== false && config[key].disabled !== true) .filter(key => config[key].display !== false && config[key].disabled !== true)
@ -1046,18 +1058,18 @@ page.prepareUploadConfig = () => {
let value let value
if (config[key].select !== undefined) { if (config[key].select !== undefined) {
if (form.elements[key].value !== 'default') if (form.elements[key].value !== 'default') {
value = form.elements[key].value value = form.elements[key].value
}
} else if (config[key].number !== undefined) { } else if (config[key].number !== undefined) {
const parsed = parseInt(form.elements[key].value) const parsed = parseInt(form.elements[key].value)
if (!isNaN(parsed) && parsed !== config[key].number.default) if (!isNaN(parsed) && parsed !== config[key].number.default) {
value = Math.min(Math.max(parsed, config[key].number.min), config[key].number.max) value = Math.min(Math.max(parsed, config[key].number.min), config[key].number.max)
}
} }
if (value !== undefined) if (value !== undefined) localStorage[lsKeys[key]] = value
localStorage[lsKeys[key]] = value else localStorage.removeItem(lsKeys[key])
else
localStorage.removeItem(lsKeys[key])
} }
swal({ swal({
@ -1104,7 +1116,7 @@ window.addEventListener('paste', event => {
}) })
window.addEventListener('DOMContentLoaded', () => { window.addEventListener('DOMContentLoaded', () => {
if (window.cookieconsent) if (window.cookieconsent) {
window.cookieconsent.initialise({ window.cookieconsent.initialise({
cookie: { cookie: {
name: 'cookieconsent_status', name: 'cookieconsent_status',
@ -1131,6 +1143,7 @@ window.addEventListener('DOMContentLoaded', () => {
href: 'cookiepolicy' href: 'cookiepolicy'
} }
}) })
}
page.checkIfPublic() page.checkIfPublic()

View File

@ -92,9 +92,11 @@ newsfeed.dismissNotification = element => {
element.parentNode.removeChild(element) element.parentNode.removeChild(element)
const keys = Object.keys(newsfeed.dismissed) const keys = Object.keys(newsfeed.dismissed)
if (keys.length > newsfeed.maxItems) if (keys.length > newsfeed.maxItems) {
for (let i = 0; i < keys.length - newsfeed.maxItems; i++) for (let i = 0; i < keys.length - newsfeed.maxItems; i++) {
delete newsfeed.dismissed[keys[i]] delete newsfeed.dismissed[keys[i]]
}
}
localStorage[newsfeed.lsKey] = JSON.stringify(newsfeed.dismissed) localStorage[newsfeed.lsKey] = JSON.stringify(newsfeed.dismissed)
} }
@ -108,8 +110,9 @@ newsfeed.do = () => {
if (items.length) { if (items.length) {
const dismissed = localStorage[newsfeed.lsKey] const dismissed = localStorage[newsfeed.lsKey]
if (dismissed) if (dismissed) {
newsfeed.dismissed = JSON.parse(dismissed) newsfeed.dismissed = JSON.parse(dismissed)
}
const element = document.createElement('section') const element = document.createElement('section')
element.id = 'newsfeed' element.id = 'newsfeed'
@ -142,11 +145,12 @@ newsfeed.do = () => {
}) })
const dismissTrigger = notificationElement.querySelector('.delete') const dismissTrigger = notificationElement.querySelector('.delete')
if (dismissTrigger) if (dismissTrigger) {
dismissTrigger.addEventListener('click', function () { dismissTrigger.addEventListener('click', function () {
event.preventDefault() event.preventDefault()
newsfeed.dismissNotification(event.target.parentNode) newsfeed.dismissNotification(event.target.parentNode)
}) })
}
column.appendChild(notificationElement) column.appendChild(notificationElement)
} }
@ -163,11 +167,13 @@ newsfeed.do = () => {
newsfeed.onloaded = () => { newsfeed.onloaded = () => {
// If the main script had already done its API check, yet newsfeed haven't been triggered, do it // If the main script had already done its API check, yet newsfeed haven't been triggered, do it
// This would only happen if this newsfeed script only gets loaded after the main script's API check // This would only happen if this newsfeed script only gets loaded after the main script's API check
if (typeof page !== 'undefined' && page.apiChecked && !newsfeed.done) if (typeof page !== 'undefined' && page.apiChecked && !newsfeed.done) {
newsfeed.do() newsfeed.do()
}
} }
if (document.readyState === 'interactive' || document.readyState === 'complete') if (document.readyState === 'interactive' || document.readyState === 'complete') {
newsfeed.onloaded() newsfeed.onloaded()
else } else {
window.addEventListener('DOMContentLoaded', () => newsfeed.onloaded()) window.addEventListener('DOMContentLoaded', () => newsfeed.onloaded())
}

View File

@ -60,8 +60,7 @@ const render = {
} }
// miku: Generate an array of file names from 001.png to 050.png // miku: Generate an array of file names from 001.png to 050.png
for (let i = 1; i <= 50; i++) for (let i = 1; i <= 50; i++) render.configs.miku.array.push(`${('00' + i).slice(-3)}.png`)
render.configs.miku.array.push(`${('00' + i).slice(-3)}.png`)
render.showTogglePrompt = () => { render.showTogglePrompt = () => {
const renderEnabled = !(localStorage[render.lsKey] === '0') const renderEnabled = !(localStorage[render.lsKey] === '0')
@ -79,11 +78,12 @@ render.showTogglePrompt = () => {
` `
const buttons = {} const buttons = {}
if (renderEnabled) if (renderEnabled) {
buttons.reload = { buttons.reload = {
text: 'Nah fam, show me a different render', text: 'Nah fam, show me a different render',
className: 'swal-button--cancel' className: 'swal-button--cancel'
} }
}
buttons.confirm = true buttons.confirm = true
swal({ swal({
@ -95,10 +95,8 @@ render.showTogglePrompt = () => {
} else if (value) { } else if (value) {
const newValue = div.querySelector('#swalRender').checked ? undefined : '0' const newValue = div.querySelector('#swalRender').checked ? undefined : '0'
if (newValue !== localStorage[render.lsKey]) { if (newValue !== localStorage[render.lsKey]) {
if (newValue) if (newValue) localStorage[render.lsKey] = newValue
localStorage[render.lsKey] = newValue else localStorage.removeItem(render.lsKey)
else
localStorage.removeItem(render.lsKey)
swal('', `Random render is now ${newValue ? 'disabled' : 'enabled'}.`, 'success', { swal('', `Random render is now ${newValue ? 'disabled' : 'enabled'}.`, 'success', {
buttons: false, buttons: false,
timer: 1500 timer: 1500
@ -111,26 +109,23 @@ render.showTogglePrompt = () => {
render.parseVersion = () => { render.parseVersion = () => {
const renderScript = document.querySelector('#renderScript') const renderScript = document.querySelector('#renderScript')
if (renderScript && renderScript.dataset.version) if (renderScript && renderScript.dataset.version) return `?v=${renderScript.dataset.version}`
return `?v=${renderScript.dataset.version}` else return ''
return ''
} }
render.do = reload => { render.do = reload => {
if (!render.done) if (!render.done) render.done = true
render.done = true
render.config = render.configs[render.type] render.config = render.configs[render.type]
if (!render.config || !render.config.array.length) if (!render.config || !render.config.array.length) return
return
const previousElement = document.querySelector('body > .render') const previousElement = document.querySelector('body > .render')
if (previousElement) if (previousElement) previousElement.remove()
previousElement.remove()
const doRender = () => { const doRender = () => {
if (render.version === undefined) if (render.version === undefined) {
render.version = render.parseVersion() render.version = render.parseVersion()
}
// Let us just allow people to get new render when toggling the option // Let us just allow people to get new render when toggling the option
render.selected = render.config.array[Math.floor(Math.random() * render.config.array.length)] render.selected = render.config.array[Math.floor(Math.random() * render.config.array.length)]
@ -158,11 +153,13 @@ render.do = reload => {
render.onloaded = () => { render.onloaded = () => {
// If the main script had already done its API check, yet render haven't been triggered, do it // If the main script had already done its API check, yet render haven't been triggered, do it
// This would only happen if this render script only gets loaded after the main script's API check // This would only happen if this render script only gets loaded after the main script's API check
if (typeof page !== 'undefined' && page.apiChecked && !render.done) if (typeof page !== 'undefined' && page.apiChecked && !render.done) {
render.do() render.do()
}
} }
if (document.readyState === 'interactive' || document.readyState === 'complete') if (document.readyState === 'interactive' || document.readyState === 'complete') {
render.onloaded() render.onloaded()
else } else {
window.addEventListener('DOMContentLoaded', () => render.onloaded()) window.addEventListener('DOMContentLoaded', () => render.onloaded())
}

View File

@ -84,17 +84,15 @@ page.getPrettyUptime = seconds => {
let minutes = Math.floor(seconds / 60) let minutes = Math.floor(seconds / 60)
seconds %= 60 seconds %= 60
if (hours < 10) if (hours < 10) hours = '0' + hours
hours = '0' + hours if (minutes < 10) minutes = '0' + minutes
if (minutes < 10) if (seconds < 10) seconds = '0' + seconds
minutes = '0' + minutes
if (seconds < 10)
seconds = '0' + seconds
if (days > 0) if (days > 0) {
return days + 'd ' + hours + ':' + minutes + ':' + seconds return days + 'd ' + hours + ':' + minutes + ':' + seconds
else } else {
return hours + ':' + minutes + ':' + seconds return hours + ':' + minutes + ':' + seconds
}
} }
page.escape = string => { page.escape = string => {
@ -103,14 +101,12 @@ page.escape = string => {
// Copyright(c) 2015 Andreas Lubbe // Copyright(c) 2015 Andreas Lubbe
// Copyright(c) 2015 Tiancheng "Timothy" Gu // Copyright(c) 2015 Tiancheng "Timothy" Gu
if (!string) if (!string) return string
return string
const str = String(string) const str = String(string)
const match = /["'&<>]/.exec(str) const match = /["'&<>]/.exec(str)
if (!match) if (!match) return str
return str
let escape let escape
let html = '' let html = ''
@ -138,8 +134,9 @@ page.escape = string => {
continue continue
} }
if (lastIndex !== index) if (lastIndex !== index) {
html += str.substring(lastIndex, index) html += str.substring(lastIndex, index)
}
lastIndex = index + 1 lastIndex = index + 1
html += escape html += escape