You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
246 lines
6.3 KiB
246 lines
6.3 KiB
/*jshint esversion: 6*/ |
|
var express = require('express'); |
|
var app = express(); |
|
var crontab = require("./crontab"); |
|
var restore = require("./restore"); |
|
var moment = require('moment'); |
|
|
|
var path = require('path'); |
|
var mime = require('mime-types'); |
|
var fs = require('fs'); |
|
var busboy = require('connect-busboy'); // for file upload |
|
|
|
// include the routes |
|
var routes = require("./routes").routes; |
|
|
|
// set the view engine to ejs |
|
app.set('view engine', 'ejs'); |
|
|
|
var bodyParser = require('body-parser'); |
|
app.use(bodyParser.json()); // to support JSON-encoded bodies |
|
app.use(bodyParser.urlencoded({ // to support URL-encoded bodies |
|
extended: true |
|
})); |
|
app.use(busboy()); // to support file uploads |
|
|
|
// include all folders |
|
app.use(express.static(__dirname + '/public')); |
|
app.use(express.static(__dirname + '/public/css')); |
|
app.use(express.static(__dirname + '/public/js')); |
|
app.use(express.static(__dirname + '/config')); |
|
app.set('views', __dirname + '/views'); |
|
|
|
// set host to 127.0.0.1 or the value set by environment var HOST |
|
app.set('host', (process.env.HOST || '127.0.0.1')); |
|
|
|
// set port to 8000 or the value set by environment var PORT |
|
app.set('port', (process.env.PORT || 8000)); |
|
|
|
// root page handler |
|
app.get(routes.root, function(req, res) { |
|
// reload the database before rendering |
|
crontab.reload_db(); |
|
// send all the required parameters |
|
crontab.crontabs( function(docs){ |
|
res.render('index', { |
|
routes : JSON.stringify(routes), |
|
crontabs : JSON.stringify(docs), |
|
backups : crontab.get_backup_names(), |
|
env : crontab.get_env(), |
|
moment: moment |
|
}); |
|
}); |
|
}); |
|
|
|
/* |
|
Handle to save crontab to database |
|
If it is a new job @param _id is set to -1 |
|
@param name, command, schedule, logging has to be sent with _id (if exists) |
|
*/ |
|
app.post(routes.save, function(req, res) { |
|
// new job |
|
if(req.body._id == -1){ |
|
crontab.create_new(req.body.name, req.body.command, req.body.schedule, req.body.logging, req.body.mailing); |
|
} |
|
// edit job |
|
else{ |
|
crontab.update(req.body); |
|
} |
|
res.end(); |
|
}); |
|
|
|
// set stop to job |
|
app.post(routes.stop, function(req, res) { |
|
crontab.status(req.body._id, true); |
|
res.end(); |
|
}); |
|
|
|
// set start to job |
|
app.post(routes.start, function(req, res) { |
|
crontab.status(req.body._id, false); |
|
res.end(); |
|
}); |
|
|
|
// remove a job |
|
app.post(routes.remove, function(req, res) { |
|
crontab.remove(req.body._id); |
|
res.end(); |
|
}); |
|
|
|
// run a job |
|
app.post(routes.run, function(req, res) { |
|
crontab.runjob(req.body._id); |
|
res.end(); |
|
}); |
|
|
|
// set crontab. Needs env_vars to be passed |
|
app.get(routes.crontab, function(req, res, next) { |
|
crontab.set_crontab(req.query.env_vars, function(err) { |
|
if (err) next(err); |
|
else res.end(); |
|
}); |
|
}); |
|
|
|
// backup crontab db |
|
app.get(routes.backup, function(req, res) { |
|
crontab.backup(); |
|
res.end(); |
|
}); |
|
|
|
// This renders the restore page similar to backup page |
|
app.get(routes.restore, function(req, res) { |
|
// get all the crontabs |
|
restore.crontabs(req.query.db, function(docs){ |
|
res.render('restore', { |
|
routes : JSON.stringify(routes), |
|
crontabs : JSON.stringify(docs), |
|
backups : crontab.get_backup_names(), |
|
db: req.query.db |
|
}); |
|
}); |
|
}); |
|
|
|
// delete backup db |
|
app.get(routes.delete_backup, function(req, res) { |
|
restore.delete(req.query.db); |
|
res.end(); |
|
}); |
|
|
|
// restore from backup db |
|
app.get(routes.restore_backup, function(req, res) { |
|
crontab.restore(req.query.db); |
|
res.end(); |
|
}); |
|
|
|
// export current crontab db so that user can download it |
|
app.get(routes.export, function(req, res) { |
|
var file = __dirname + '/crontabs/crontab.db'; |
|
|
|
var filename = path.basename(file); |
|
var mimetype = mime.lookup(file); |
|
|
|
res.setHeader('Content-disposition', 'attachment; filename=' + filename); |
|
res.setHeader('Content-type', mimetype); |
|
|
|
var filestream = fs.createReadStream(file); |
|
filestream.pipe(res); |
|
}); |
|
|
|
// import from exported crontab db |
|
app.post(routes.import, function(req, res) { |
|
var fstream; |
|
req.pipe(req.busboy); |
|
req.busboy.on('file', function (fieldname, file, filename) { |
|
fstream = fs.createWriteStream(__dirname + '/crontabs/crontab.db'); |
|
file.pipe(fstream); |
|
fstream.on('close', function () { |
|
crontab.reload_db(); |
|
res.redirect(routes.root); |
|
}); |
|
}); |
|
}); |
|
|
|
// import from current ACTUALL crontab |
|
app.get(routes.import_crontab, function(req, res) { |
|
crontab.import_crontab(); |
|
res.end(); |
|
}); |
|
|
|
// get the log file a given job. id passed as query param |
|
app.get(routes.logger, function(req, res) { |
|
_file = crontab.log_folder +"/"+req.query.id+".log"; |
|
if (fs.existsSync(_file)) |
|
res.sendFile(_file); |
|
else |
|
res.end("No errors logged yet"); |
|
}); |
|
|
|
// error handler |
|
app.use(function(err, req, res, next) { |
|
var data = {}; |
|
var statusCode = err.statusCode || 500; |
|
|
|
data.message = err.message || 'Internal Server Error'; |
|
|
|
if (process.env.NODE_ENV === 'development' && err.stack) { |
|
data.stack = err.stack; |
|
} |
|
|
|
if (parseInt(data.statusCode) >= 500) { |
|
console.error(err); |
|
} |
|
|
|
res.status(statusCode).json(data); |
|
}); |
|
|
|
process.on('SIGINT', function() { |
|
console.log("Exiting crontab-ui"); |
|
process.exit(); |
|
}) |
|
|
|
process.on('SIGTERM', function() { |
|
console.log("Exiting crontab-ui"); |
|
process.exit(); |
|
}) |
|
|
|
app.listen(app.get('port'), app.get('host'), function() { |
|
console.log("Node version:", process.versions.node); |
|
fs.access(__dirname + "/crontabs/", fs.W_OK, function(err) { |
|
if(err){ |
|
console.error("Write access to", __dirname + "/crontabs/", "DENIED."); |
|
process.exit(1); |
|
} |
|
}); |
|
// If --autosave is used then we will also save whatever is in the db automatically without having to mention it explictly |
|
// we do this by watching log file and setting a on change hook to it |
|
if (process.argv.includes("--autosave")){ |
|
crontab.autosave_crontab(()=>{}); |
|
fs.watchFile(__dirname + '/crontabs/crontab.db', () => { |
|
crontab.autosave_crontab(()=>{ |
|
console.log("Attempted to autosave crontab"); |
|
}); |
|
}); |
|
} |
|
if (process.argv.includes("--reset")){ |
|
console.log("Resetting crontab-ui"); |
|
var crontabdb = __dirname + "/crontabs/crontab.db"; |
|
var envdb = __dirname + "/crontabs/env.db"; |
|
|
|
console.log("Deleting " + crontabdb); |
|
try{ |
|
fs.unlinkSync(crontabdb); |
|
} catch (e) { |
|
console.log("Unable to delete " + crontabdb); |
|
} |
|
|
|
console.log("Deleting " + envdb); |
|
try{ |
|
fs.unlinkSync(envdb); |
|
} catch (e) { |
|
console.log("Unable to delete " + envdb); |
|
} |
|
|
|
crontab.reload_db(); |
|
} |
|
console.log("Crontab UI is running at http://" + app.get('host') + ":" + app.get('port')); |
|
});
|
|
|