376 lines
14 KiB
JavaScript
376 lines
14 KiB
JavaScript
const path = require('path');
|
|
const fs = require('fs');
|
|
const express = require('express');
|
|
const http = require('http');
|
|
const socketio = require('socket.io');
|
|
const helmet = require('helmet');
|
|
const bodyParser = require('body-parser');
|
|
const fileUpload = require('express-fileupload');
|
|
const filenamify = require('filenamify');
|
|
const unusedFilename = require('unused-filename');
|
|
const striptags = require('striptags');
|
|
const snarkdown = require('snarkdown');
|
|
const fecha = require('fecha');
|
|
|
|
const settings = require('./settings.json');
|
|
|
|
function Server () {
|
|
this.server = express();
|
|
this.http = http.Server(this.server);
|
|
this.io = socketio(this.http);
|
|
|
|
this.fileLocation = path.resolve(settings.fileLocation);
|
|
this.historyLocation = path.resolve(settings.historyLocation);
|
|
|
|
this.templateCache = {};
|
|
|
|
this.takenBooks = [];
|
|
|
|
this.server.use(helmet());
|
|
|
|
this.server.use(bodyParser.json()); // support json encoded bodies
|
|
this.server.use(bodyParser.urlencoded({ extended: true })); // support encoded bodies
|
|
|
|
this.server.use(fileUpload({ // support file uploads
|
|
limits: {
|
|
fileSize: (settings.maxFileSize > 0 ? settings.maxFileSize * 1024 * 1024 : Infinity), // filesize in bytes (settings accepts MB)
|
|
},
|
|
}));
|
|
|
|
this.server.use('/files', express.static(path.join(__dirname, './public/files/')));
|
|
this.server.use('/css', express.static(path.resolve('./node_modules/bulma/css/')));
|
|
this.server.use('/css', express.static(path.join(__dirname, './public/css/')));
|
|
this.server.use('/js', express.static(path.join(__dirname, './public/js/')));
|
|
this.server.use('/js', express.static(path.resolve('./node_modules/jquery/dist/')));
|
|
this.server.use('/js', express.static(path.resolve('./node_modules/socket.io-client/dist/')));
|
|
|
|
this.server.get('/', (req, res) => {
|
|
const html = this.generateHomePage(req);
|
|
if (html) {
|
|
res.send(html);
|
|
} else {
|
|
res.send('Something went wrong!');
|
|
}
|
|
});
|
|
|
|
this.server.get('/give', (req, res) => {
|
|
const resourcePath = (req.url.substr(-1) === '/' ? '../' : './');
|
|
const body = this.fillTemplate('./templates/pages/uploadForm.html', { resourcePath });
|
|
const html = this.fillTemplate('./templates/htmlContainer.html', { title: 'Give a Book', resourcePath, body });
|
|
res.send(html);
|
|
});
|
|
this.server.post('/give', (req, res) => {
|
|
const resourcePath = (req.url.substr(-1) === '/' ? '../' : './');
|
|
const { title, author, summary, contributor } = req.body;
|
|
if (Object.keys(req.files).length > 0
|
|
&& req.body.hasOwnProperty('title') && title.trim() !== ''
|
|
&& req.body.hasOwnProperty('summary') && summary.trim() !== '') {
|
|
const { book } = req.files;
|
|
const fileType = book.name.substr(book.name.lastIndexOf('.'));
|
|
this.addBook({ book, title, author, summary, contributor, fileType }, () => {
|
|
const messageBox = this.fillTemplate('./templates/elements/messageBox.html', {
|
|
style: 'is-success',
|
|
header: 'Upload Successful',
|
|
message: 'Thank you for your contribution!'
|
|
});
|
|
const modal = this.fillTemplate('./templates/elements/modal.html', {
|
|
isActive: 'is-active',
|
|
content: messageBox,
|
|
});
|
|
const body = this.fillTemplate('./templates/pages/uploadForm.html', { resourcePath });
|
|
const html = this.fillTemplate('./templates/htmlContainer.html', { title: 'Give a Book', resourcePath, body, modal });
|
|
res.send(html);
|
|
}, (err) => {
|
|
const messageBox = this.fillTemplate('./templates/elements/messageBox.html', {
|
|
style: 'is-danger',
|
|
header: 'Upload Failed',
|
|
message: err,
|
|
});
|
|
const modal = this.fillTemplate('./templates/elements/modal.html', {
|
|
isActive: 'is-active',
|
|
content: messageBox,
|
|
});
|
|
const body = this.fillTemplate('./templates/pages/uploadForm.html', { resourcePath, title, author, summary, contributor });
|
|
const html = this.fillTemplate('./templates/htmlContainer.html', { title: 'Give a Book', resourcePath, body, modal });
|
|
res.send(html);
|
|
});
|
|
} else {
|
|
let errorMessage = '';
|
|
if (Object.keys(req.files).length <= 0) {
|
|
errorMessage += 'You have not selected a file.';
|
|
}
|
|
if (!req.body.hasOwnProperty('title') || req.body.title.trim() === '') {
|
|
errorMessage += (errorMessage.length > 0 ? '<br>' : '') + 'You have not written a title.';
|
|
}
|
|
if (!req.body.hasOwnProperty('summary') || req.body.summary.trim() === '') {
|
|
errorMessage += (errorMessage.length > 0 ? '<br>' : '') + 'You have not written a summary.';
|
|
}
|
|
const message = this.fillTemplate('./templates/elements/messageBox.html', {
|
|
style: 'is-danger',
|
|
header: 'Missing Required Fields',
|
|
message: errorMessage,
|
|
});
|
|
const body = this.fillTemplate('./templates/pages/uploadForm.html', { resourcePath, title, author, summary, contributor });
|
|
const html = this.fillTemplate('./templates/htmlContainer.html', { title: 'Give a Book', resourcePath, body, message });
|
|
res.send(html);
|
|
}
|
|
});
|
|
|
|
this.server.get('/history', (req, res) => {
|
|
const html = this.generateHistoryPage(req);
|
|
if (html) {
|
|
res.send(html);
|
|
} else {
|
|
res.send('Something went wrong!');
|
|
}
|
|
});
|
|
|
|
this.server.get('/about', (req, res) => {
|
|
const body = this.fillTemplate('./templates/pages/about.html');
|
|
const html = this.fillTemplate('./templates/htmlContainer.html', { title: 'About', body });
|
|
if (html) {
|
|
res.send(html);
|
|
} else {
|
|
res.send('Something went wrong!');
|
|
}
|
|
});
|
|
|
|
this.io.on('connection', socket => {
|
|
this.broadcastVisitors();
|
|
|
|
socket.on('take book', bookId => {
|
|
const fileLocation = this.takeBook(bookId, socket.id);
|
|
if (fileLocation) {
|
|
console.log(socket.id + ' removed ' + bookId);
|
|
const downloadLocation = fileLocation.substr(fileLocation.lastIndexOf('/'));
|
|
socket.emit('get book', encodeURI('./files' + downloadLocation));
|
|
}
|
|
});
|
|
|
|
socket.on('disconnect', () => {
|
|
this.broadcastVisitors();
|
|
this.deleteBooks(socket.id);
|
|
});
|
|
});
|
|
}
|
|
|
|
Server.prototype.fillTemplate = function (file, templateVars = {}) {
|
|
let data;
|
|
if (this.templateCache.hasOwnProperty(file)) {
|
|
data = this.templateCache[file];
|
|
} else {
|
|
data = fs.readFileSync(path.join(__dirname, file), 'utf8');
|
|
}
|
|
if (data) {
|
|
if (!this.templateCache.hasOwnProperty(file)) {
|
|
this.templateCache[file] = data;
|
|
}
|
|
|
|
let filledTemplate = data.replace(/\{\{siteTitle\}\}/g, settings.siteTitle)
|
|
.replace(/\{\{titleSeparator\}\}/g, settings.titleSeparator)
|
|
.replace(/\{\{allowedFormats\}\}/g, settings.allowedFormats.join(','))
|
|
.replace(/\{\{maxFileSize\}\}/g, (settings.maxFileSize > 0 ? settings.maxFileSize + 'MB' : 'no'));
|
|
|
|
for (let templateVar in templateVars) {
|
|
const regExp = new RegExp('\{\{' + templateVar + '\}\}', 'g')
|
|
filledTemplate = filledTemplate.replace(regExp, templateVars[templateVar]);
|
|
}
|
|
|
|
// If any template variable is not provided, don't even render them.
|
|
filledTemplate = filledTemplate.replace(/\{\{[a-zA-Z0-9\-_]+\}\}/g, '');
|
|
|
|
return filledTemplate;
|
|
}
|
|
|
|
return data;
|
|
}
|
|
|
|
Server.prototype.generateHomePage = function (req) {
|
|
const files = fs.readdirSync(this.fileLocation).filter(fileName => fileName.includes('.json'));
|
|
let books = files.map(fileName => {
|
|
const bookData = JSON.parse(fs.readFileSync(path.resolve(this.fileLocation, fileName), 'utf8'));
|
|
if (bookData.hasOwnProperty('fileName')) return '';
|
|
|
|
const id = fileName.replace('.json', '');
|
|
const confirmId = 'confirm_' + id;
|
|
const added = fecha.format(new Date(bookData.added), 'hh:mm:ssA on dddd MMMM Do, YYYY');
|
|
const modal = this.fillTemplate('./templates/elements/modalCard.html', {
|
|
id,
|
|
header: '<h2 class="title">' + bookData.title + '</h2><h4 class="subtitle">' + bookData.author + '</h4>',
|
|
content: this.fillTemplate('./templates/elements/bookInfo.html', {
|
|
contributor: bookData.contributor,
|
|
fileFormat: bookData.fileType,
|
|
added,
|
|
summary: snarkdown(bookData.summary),
|
|
})
|
|
+ this.fillTemplate('./templates/elements/modal.html', {
|
|
id: confirmId,
|
|
content: this.fillTemplate('./templates/elements/messageBox.html', {
|
|
header: 'Download Your Book',
|
|
message: this.fillTemplate('./templates/elements/takeConfirm.html', { id }),
|
|
}),
|
|
}),
|
|
footer: '<a class="button close">Close</a> <a class="button is-success modal-button" data-modal="' + confirmId + '">Take Book</a>',
|
|
});
|
|
return this.fillTemplate('./templates/elements/book.html', {
|
|
id,
|
|
title: bookData.title,
|
|
author: bookData.author,
|
|
fileType: bookData.fileType,
|
|
modal,
|
|
});
|
|
}).join('');
|
|
|
|
if (books == '') {
|
|
books = '<div class="column"><div class="content">The shelf is empty. Would you like to <a href="/give">add a book</a>?</div></div>';
|
|
}
|
|
|
|
const body = '<h2 class="title">Available Books</h2><div class="columns is-multiline">' + books + '</div>';
|
|
return this.fillTemplate('./templates/htmlContainer.html', {
|
|
title: 'View',
|
|
resourcePath: (req.url.substr(-1) === '/' ? '../' : './'),
|
|
body
|
|
});
|
|
}
|
|
|
|
Server.prototype.generateHistoryPage = function (req) {
|
|
const files = fs.readdirSync(this.historyLocation).filter(fileName => fileName.includes('.json'));
|
|
let history = files.map(fileName => {
|
|
const bookData = JSON.parse(fs.readFileSync(path.resolve(this.historyLocation, fileName), 'utf8'));
|
|
const id = fileName.replace('.json', '');
|
|
const added = fecha.format(new Date(bookData.added), 'hh:mm:ssA on dddd MMMM Do, YYYY');
|
|
const removed = fecha.format(new Date(parseInt(id)), 'hh:mm:ssA on dddd MMMM Do, YYYY');
|
|
const removedTag = '<div class="control"><div class="tags has-addons"><span class="tag">Taken</span><span class="tag is-primary">' + removed + '</span></div></div>';
|
|
const modal = this.fillTemplate('./templates/elements/modalCard.html', {
|
|
id,
|
|
header: '<h2 class="title">' + bookData.title + '</h2><h4 class="subtitle">' + bookData.author + '</h4>',
|
|
content: this.fillTemplate('./templates/elements/bookInfo.html', {
|
|
contributor: bookData.contributor,
|
|
fileFormat: bookData.fileType,
|
|
added,
|
|
removedTag,
|
|
summary: snarkdown(bookData.summary),
|
|
}),
|
|
footer: '<a class="button close">Close</a>',
|
|
});
|
|
return this.fillTemplate('./templates/elements/book.html', {
|
|
id,
|
|
title: bookData.title,
|
|
author: bookData.author,
|
|
fileType: bookData.fileType,
|
|
modal,
|
|
});
|
|
}).join('');
|
|
|
|
if (history == '') {
|
|
history = '<div class="column"><div class="content">No books have been taken yet. Would you like to <a href="/">take a book</a>?</div></div>';
|
|
}
|
|
|
|
const body = '<h2 class="title">History</h2><div class="columns is-multiline">' + history + '</div>';
|
|
return this.fillTemplate('./templates/htmlContainer.html', {
|
|
title: 'History',
|
|
resourcePath: (req.url.substr(-1) === '/' ? '../' : './'),
|
|
body
|
|
});
|
|
}
|
|
|
|
Server.prototype.broadcastVisitors = function () {
|
|
const numberConnected = this.io.of('/').clients().connected.length;
|
|
this.io.emit('connected', numberConnected);
|
|
}
|
|
|
|
Server.prototype.start = function () {
|
|
this.http.listen(settings.port, () => {
|
|
console.log('Started server on port ' + settings.port);
|
|
});
|
|
}
|
|
|
|
Server.prototype.addBook = function (uploadData = {}, success = () => {}, error = () => {}) {
|
|
const { book } = uploadData;
|
|
|
|
// If the file is too big, error out.
|
|
if (book.truncated === true) {
|
|
delete book;
|
|
return error('The file provided is too big');
|
|
}
|
|
|
|
const bookId = this.uuid4();
|
|
const bookPath = path.resolve(this.fileLocation, bookId);
|
|
|
|
const bookData = {
|
|
title: striptags(uploadData.title.trim()),
|
|
author: striptags(uploadData.author.trim()),
|
|
summary: striptags(uploadData.summary.trim().replace(/\r\n/g, '\n')),
|
|
contributor: striptags(uploadData.contributor.trim()),
|
|
added: Date.now(),
|
|
fileType: book.name.substr(book.name.lastIndexOf('.')),
|
|
}
|
|
|
|
const bookFilePath = unusedFilename.sync(path.resolve(bookPath + bookData.fileType));
|
|
return book.mv(bookFilePath, function (err) {
|
|
if (err) {
|
|
console.log(err);
|
|
error(err);
|
|
} else {
|
|
const bookDataPath = unusedFilename.sync(path.resolve(bookPath + '.json'));
|
|
fs.writeFileSync(bookDataPath, JSON.stringify(bookData));
|
|
success();
|
|
// console.log('uploaded ' + bookData.title + ' to ' + bookFilePath + ', and saved metadata to ' + bookDataPath);
|
|
}
|
|
});
|
|
}
|
|
|
|
Server.prototype.takeBook = function (bookId, socketId) {
|
|
return this.checkId(bookId, (bookPath, bookDataPath, bookData) => {
|
|
const bookName = filenamify(bookData.title);
|
|
const newFileName = unusedFilename.sync(path.resolve(this.fileLocation, bookName + bookData.fileType));
|
|
bookData.fileName = newFileName;
|
|
fs.renameSync(bookPath, newFileName);
|
|
fs.writeFileSync(bookDataPath, JSON.stringify(bookData));
|
|
this.takenBooks.push({ socketId, bookId });
|
|
return newFileName.replace(/\\/g, '/');
|
|
});
|
|
}
|
|
|
|
Server.prototype.checkId = function (bookId, callback = () => {}) {
|
|
const bookDataPath = path.resolve(this.fileLocation, bookId + '.json');
|
|
if (fs.existsSync(bookDataPath)) {
|
|
const bookDataRaw = fs.readFileSync(bookDataPath);
|
|
if (bookDataRaw) {
|
|
const bookData = JSON.parse(bookDataRaw);
|
|
const bookPath = bookData.hasOwnProperty('fileName') ? bookData.fileName : path.resolve(this.fileLocation, bookId + bookData.fileType);
|
|
if (fs.existsSync(bookPath)) {
|
|
return callback(bookPath, bookDataPath, bookData);
|
|
}
|
|
}
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
Server.prototype.deleteBooks = function (socketId) {
|
|
this.takenBooks.forEach(data => {
|
|
if (data.socketId === socketId) {
|
|
const check = this.checkId(data.bookId, (bookPath, bookDataPath) => {
|
|
fs.unlinkSync(bookPath);
|
|
fs.renameSync(bookDataPath, unusedFilename.sync(path.resolve(this.historyLocation, Date.now() + '.json')));
|
|
});
|
|
if (check === false) {
|
|
console.log('couldn\'t find data.bookId');
|
|
}
|
|
}
|
|
});
|
|
this.takenBooks = this.takenBooks.filter(data => data.socketId === socketId);
|
|
}
|
|
|
|
Server.prototype.uuid4 = function () {
|
|
// https://stackoverflow.com/a/2117523
|
|
return 'xxxxxxxx-xxxx-4xxx-yxxx-xxxxxxxxxxxx'.replace(/[xy]/g, function (c) {
|
|
var r = Math.random() * 16 | 0, v = c == 'x' ? r : (r & 0x3 | 0x8);
|
|
return v.toString(16);
|
|
});
|
|
}
|
|
|
|
const server = new Server();
|
|
server.start();
|