|
|
import fs from 'node:fs'; |
|
|
import path from 'node:path'; |
|
|
import { Buffer } from 'node:buffer'; |
|
|
|
|
|
import express from 'express'; |
|
|
import sanitize from 'sanitize-filename'; |
|
|
|
|
|
import { clientRelativePath, removeFileExtension, getImages, isPathUnderParent } from '../util.js'; |
|
|
import { MEDIA_EXTENSIONS } from '../constants.js'; |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
function ensureDirectoryExistence(filePath) { |
|
|
const dirname = path.dirname(filePath); |
|
|
if (fs.existsSync(dirname)) { |
|
|
return true; |
|
|
} |
|
|
ensureDirectoryExistence(dirname); |
|
|
fs.mkdirSync(dirname); |
|
|
} |
|
|
|
|
|
export const router = express.Router(); |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
router.post('/upload', async (request, response) => { |
|
|
try { |
|
|
if (!request.body) { |
|
|
return response.status(400).send({ error: 'No data provided' }); |
|
|
} |
|
|
|
|
|
const { image, format } = request.body; |
|
|
|
|
|
if (!image) { |
|
|
return response.status(400).send({ error: 'No image data provided' }); |
|
|
} |
|
|
|
|
|
const validFormat = MEDIA_EXTENSIONS.includes(format); |
|
|
if (!validFormat) { |
|
|
return response.status(400).send({ error: 'Invalid image format' }); |
|
|
} |
|
|
|
|
|
|
|
|
let filename; |
|
|
if (request.body.filename) { |
|
|
filename = `${removeFileExtension(request.body.filename)}.${format}`; |
|
|
} else { |
|
|
filename = `${Date.now()}.${format}`; |
|
|
} |
|
|
|
|
|
|
|
|
let pathToNewFile = path.join(request.user.directories.userImages, sanitize(filename)); |
|
|
if (request.body.ch_name) { |
|
|
pathToNewFile = path.join(request.user.directories.userImages, sanitize(request.body.ch_name), sanitize(filename)); |
|
|
} |
|
|
|
|
|
ensureDirectoryExistence(pathToNewFile); |
|
|
const imageBuffer = Buffer.from(image, 'base64'); |
|
|
await fs.promises.writeFile(pathToNewFile, new Uint8Array(imageBuffer)); |
|
|
response.send({ path: clientRelativePath(request.user.directories.root, pathToNewFile) }); |
|
|
} catch (error) { |
|
|
console.error(error); |
|
|
response.status(500).send({ error: 'Failed to save the image' }); |
|
|
} |
|
|
}); |
|
|
|
|
|
router.post('/list/:folder?', (request, response) => { |
|
|
try { |
|
|
if (request.params.folder) { |
|
|
if (request.body.folder) { |
|
|
return response.status(400).send({ error: 'Folder specified in both URL and body' }); |
|
|
} |
|
|
|
|
|
console.warn('Deprecated: Use POST /api/images/list with folder in request body'); |
|
|
request.body.folder = request.params.folder; |
|
|
} |
|
|
|
|
|
if (!request.body.folder) { |
|
|
return response.status(400).send({ error: 'No folder specified' }); |
|
|
} |
|
|
|
|
|
const directoryPath = path.join(request.user.directories.userImages, sanitize(request.body.folder)); |
|
|
const sort = request.body.sortField || 'date'; |
|
|
const order = request.body.sortOrder || 'asc'; |
|
|
|
|
|
if (!fs.existsSync(directoryPath)) { |
|
|
fs.mkdirSync(directoryPath, { recursive: true }); |
|
|
} |
|
|
|
|
|
const images = getImages(directoryPath, sort); |
|
|
if (order === 'desc') { |
|
|
images.reverse(); |
|
|
} |
|
|
return response.send(images); |
|
|
} catch (error) { |
|
|
console.error(error); |
|
|
return response.status(500).send({ error: 'Unable to retrieve files' }); |
|
|
} |
|
|
}); |
|
|
|
|
|
router.post('/folders', (request, response) => { |
|
|
try { |
|
|
const directoryPath = request.user.directories.userImages; |
|
|
if (!fs.existsSync(directoryPath)) { |
|
|
fs.mkdirSync(directoryPath, { recursive: true }); |
|
|
} |
|
|
|
|
|
const folders = fs.readdirSync(directoryPath, { withFileTypes: true }) |
|
|
.filter(dirent => dirent.isDirectory()) |
|
|
.map(dirent => dirent.name); |
|
|
|
|
|
return response.send(folders); |
|
|
} catch (error) { |
|
|
console.error(error); |
|
|
return response.status(500).send({ error: 'Unable to retrieve folders' }); |
|
|
} |
|
|
}); |
|
|
|
|
|
router.post('/delete', async (request, response) => { |
|
|
try { |
|
|
if (!request.body.path) { |
|
|
return response.status(400).send('No path specified'); |
|
|
} |
|
|
|
|
|
const pathToDelete = path.join(request.user.directories.root, request.body.path); |
|
|
if (!isPathUnderParent(request.user.directories.userImages, pathToDelete)) { |
|
|
return response.status(400).send('Invalid path'); |
|
|
} |
|
|
|
|
|
if (!fs.existsSync(pathToDelete)) { |
|
|
return response.status(404).send('File not found'); |
|
|
} |
|
|
|
|
|
fs.unlinkSync(pathToDelete); |
|
|
console.info(`Deleted image: ${request.body.path} from ${request.user.profile.handle}`); |
|
|
return response.sendStatus(200); |
|
|
} catch (error) { |
|
|
console.error(error); |
|
|
return response.sendStatus(500); |
|
|
} |
|
|
}); |
|
|
|