/*jshint esversion: 6*/ //load database var Datastore = require('nedb'); var path = require("path"); exports.db_folder = process.env.CRON_DB_PATH === undefined ? path.join(__dirname, "crontabs") : process.env.CRON_DB_PATH; console.log("Cron db path: " + exports.db_folder); exports.log_folder = path.join(exports.db_folder, 'logs'); exports.env_file = path.join(exports.db_folder, 'env.db'); exports.crontab_db_file = path.join(exports.db_folder, 'crontab.db'); var db = new Datastore({ filename: exports.crontab_db_file}); var cronPath = "/tmp"; if(process.env.CRON_PATH !== undefined) { console.log(`Path to crond files set using env variables ${process.env.CRON_PATH}`); cronPath = process.env.CRON_PATH; } db.loadDatabase(function (err) { if (err) throw err; // no hope, just terminate }); var exec = require('child_process').exec; var fs = require('fs'); var cron_parser = require("cron-parser"); var cronstrue = require('cronstrue/i18n'); var humanCronLocate = process.env.HUMANCRON ?? "en" if (!fs.existsSync(exports.log_folder)){ fs.mkdirSync(exports.log_folder); } crontab = function(name, command, schedule, stopped, logging, mailing){ var data = {}; data.name = name; data.command = command; data.schedule = schedule; if(stopped !== null) { data.stopped = stopped; } data.timestamp = (new Date()).toString(); data.logging = logging; if (!mailing) mailing = {}; data.mailing = mailing; return data; }; exports.create_new = function(name, command, schedule, logging, mailing){ var tab = crontab(name, command, schedule, false, logging, mailing); tab.created = new Date().valueOf(); tab.saved = false; db.insert(tab); }; exports.update = function(data){ var tab = crontab(data.name, data.command, data.schedule, null, data.logging, data.mailing); tab.saved = false; db.update({_id: data._id}, tab); }; exports.status = function(_id, stopped){ db.update({_id: _id},{$set: {stopped: stopped, saved: false}}); }; exports.remove = function(_id){ db.remove({_id: _id}, {}); }; // Iterates through all the crontab entries in the db and calls the callback with the entries exports.crontabs = function(callback){ db.find({}).sort({ created: -1 }).exec(function(err, docs){ for(var i=0; i&1 1>&2 2>&3 | tee " + stderr + ") 3>&1 1>&2 2>&3" crontab_job_string = "(" + crontab_job_string + ")" if (tab.logging && tab.logging == "true") { crontab_job_string += "; if test -f " + stderr + "; then date >> \"" + log_file + "\"" + "; cat " + stderr + " >> \"" + log_file + "\"" + "; fi"; crontab_job_string += "; if test -f " + stdout + "; then date >> \"" + log_file_stdout + "\"" + "; cat " + stdout + " >> \"" + log_file_stdout + "\"" + "; fi"; } if (tab.hook) { crontab_job_string += "; if test -f " + stdout + "; then " + tab.hook + " < " + stdout + "; fi"; } if (tab.mailing && JSON.stringify(tab.mailing) != "{}"){ crontab_job_string += "; /usr/local/bin/node " + __dirname + "/bin/crontab-ui-mailer.js " + tab._id + " " + stdout + " " + stderr; } return crontab_job_string; } add_env_vars = function(env_vars, command) { console.log("env vars"); console.log(env_vars) if (env_vars) return "(" + env_vars.replace(/\s*\n\s*/g,' ').trim() + "; (" + command + "))"; return command; } // Set actual crontab file from the db exports.set_crontab = function(env_vars, callback) { exports.crontabs( function(tabs){ var crontab_string = ""; if (env_vars) { crontab_string += env_vars; crontab_string += "\n"; } tabs.forEach(function(tab){ if(!tab.stopped) { crontab_string += tab.schedule crontab_string += " " crontab_string += make_command(tab) crontab_string += "\n"; } }); fs.writeFile(exports.env_file, env_vars, function(err) { if (err) { console.error(err); callback(err); } // In docker we're running as the root user, so we need to write the file as root and not crontab var fileName = process.env.CRON_IN_DOCKER !== undefined ? "root" : "crontab"; fs.writeFile(path.join(cronPath, fileName), crontab_string, function(err) { if (err) { console.error(err); return callback(err); } exec("crontab " + path.join(cronPath, fileName), function(err) { if (err) { console.error(err); return callback(err); } else { db.update({},{$set: {saved: true}}, {multi: true}); callback(); } }); }); }); }); }; exports.get_backup_names = function(){ var backups = []; fs.readdirSync(exports.db_folder).forEach(function(file){ // file name begins with backup if(file.indexOf("backup") === 0){ backups.push(file); } }); let backup_date = (backup_name) => { let T = backup_name.split("backup")[1]; return new Date(T.substring(0, T.length-3)).valueOf(); } backups.sort((a, b) => backup_date(b) - backup_date(a)); return backups; }; exports.backup = (callback) => { fs.copyFile(exports.crontab_db_file, path.join(exports.db_folder, 'backup ' + (new Date()).toString().replace("+", " ") + '.db'), (err) => { if (err) { console.error(err); return callback(err); } callback(); }); }; exports.restore = function(db_name){ fs.createReadStream(path.join(exports.db_folder, db_name)).pipe(fs.createWriteStream(exports.crontab_db_file)); db.loadDatabase(); // reload the database }; exports.reload_db = function(){ db.loadDatabase(); }; exports.get_env = function(){ if (fs.existsSync(exports.env_file)) { return fs.readFileSync(exports.env_file , 'utf8').replace("\n", "\n"); } return ""; }; exports.import_crontab = function(){ exec("crontab -l", function(error, stdout, stderr){ var lines = stdout.split("\n"); var namePrefix = new Date().getTime(); lines.forEach(function(line, index){ line = line.replace(/\t+/g, ' '); var regex = /^((\@[a-zA-Z]+\s+)|(([^\s]+)\s+([^\s]+)\s+([^\s]+)\s+([^\s]+)\s+([^\s]+)\s+))/; var command = line.replace(regex, '').trim(); var schedule = line.replace(command, '').trim(); var is_valid = false; try { is_valid = cron_parser.parseString(line).expressions.length > 0; } catch (e){} if(command && schedule && is_valid){ var name = namePrefix + '_' + index; db.findOne({ command: command, schedule: schedule }, function(err, doc) { if(err) { throw err; } if(!doc){ exports.create_new(name, command, schedule, null); } else{ doc.command = command; doc.schedule = schedule; exports.update(doc); } }); } }); }); }; exports.autosave_crontab = function(callback) { let env_vars = exports.get_env(); exports.set_crontab(env_vars, callback); };