/* Copyright (C) 2004 J.F.Dockes * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 2 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program; if not, write to the * Free Software Foundation, Inc., * 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. */ #ifndef TEST_ECRONTAB #include "autoconfig.h" #include #include "ecrontab.h" #include "execmd.h" #include "smallut.h" #include "debuglog.h" // Read crontab file and split it into lines. static bool eCrontabGetLines(vector& lines) { string crontab; ExecCmd croncmd; vector args; int status; // Retrieve current crontab contents. An error here means that no // crontab exists, and is not fatal, but we return a different // status than for an empty one args.push_back("-l"); if ((status = croncmd.doexec("crontab", args, 0, &crontab))) { lines.clear(); return false; } // Split crontab into lines stringToTokens(crontab, lines, "\n"); return true; } // Concatenate lines and write crontab static bool eCrontabWriteFile(const vector& lines, string& reason) { string crontab; ExecCmd croncmd; vector args; int status; for (vector::const_iterator it = lines.begin(); it != lines.end(); it++) { crontab += *it + "\n"; } args.push_back("-"); if ((status = croncmd.doexec("crontab", args, &crontab, 0))) { char nbuf[30]; sprintf(nbuf, "0x%x", status); reason = string("Exec crontab -l failed: status: ") + nbuf; return false; } return true; } // Add / change / delete entry identified by marker and id bool editCrontab(const string& marker, const string& id, const string& sched, const string& cmd, string& reason) { vector lines; if (!eCrontabGetLines(lines)) { // Special case: cmd is empty, no crontab, don't create one if (cmd.empty()) return true; } // Remove old copy if any for (vector::iterator it = lines.begin(); it != lines.end(); it++) { // Skip comment if (it->find_first_of("#") == it->find_first_not_of(" \t")) continue; if (it->find(marker) != string::npos && it->find(id) != string::npos) { lines.erase(it); break; } } if (!cmd.empty()) { string nline = sched + " " + marker + " " + id + " " + cmd; lines.push_back(nline); } if (!eCrontabWriteFile(lines, reason)) return false; return true; } bool checkCrontabUnmanaged(const string& marker, const string& data) { vector lines; if (!eCrontabGetLines(lines)) { // No crontab, answer is no return false; } // Scan crontab for (vector::iterator it = lines.begin(); it != lines.end(); it++) { if (it->find(marker) == string::npos && it->find(data) != string::npos) { return true; } } return false; } /** Retrieve the scheduling for a crontab entry */ bool getCrontabSched(const string& marker, const string& id, vector& sched) { LOGDEB0(("getCrontabSched: marker[%s], id[%s]\n", marker.c_str(), id.c_str())); vector lines; if (!eCrontabGetLines(lines)) { // No crontab, answer is no sched.clear(); return false; } string line; for (vector::iterator it = lines.begin(); it != lines.end(); it++) { // Skip comment if (it->find_first_of("#") == it->find_first_not_of(" \t")) continue; if (it->find(marker) != string::npos && it->find(id) != string::npos) { line = *it; break; } } stringToTokens(line, sched); sched.resize(5); return true; } #else // TEST -> #include #include #include #include #include #include #include using namespace std; #include "ecrontab.h" static char *thisprog; static char usage [] = " -a add or replace crontab line \n" " -d delete crontab line \n" " -s get scheduling \n" " -c check for unmanaged lines for string\n" ; static void Usage(void) { fprintf(stderr, "%s: usage:\n%s", thisprog, usage); exit(1); } static int op_flags; #define OPT_MOINS 0x1 #define OPT_a 0x2 #define OPT_d 0x4 #define OPT_w 0x8 #define OPT_c 0x10 #define OPT_s 0x20 const string& marker("RCLCRON_RCLINDEX="); // Note of course the -w does not make sense for a cron entry const string& cmd0("recollindex -w "); const string& id("RECOLL_CONFDIR=\"/home/dockes/.recoll/\""); const string& sched("30 8 * 1 *"); int main(int argc, char **argv) { thisprog = argv[0]; argc--; argv++; string wt = "10"; string cmd; while (argc > 0 && **argv == '-') { (*argv)++; if (!(**argv)) /* Cas du "adb - core" */ Usage(); while (**argv) switch (*(*argv)++) { case 'a': op_flags |= OPT_a; break; case 'c': op_flags |= OPT_c; if (argc < 2) Usage(); cmd = *(++argv); argc--; goto b1; case 'd': op_flags |= OPT_d; break; case 's': op_flags |= OPT_s; break; case 'w': op_flags |= OPT_w; if (argc < 2) Usage(); wt = *(++argv); argc--; goto b1; default: Usage(); break; } b1: argc--; argv++; } if (argc != 0) Usage(); string reason; bool status = false; if (op_flags & OPT_a) { cmd = cmd0 + wt; status = editCrontab(marker, id, sched, cmd, reason); } else if (op_flags & OPT_d) { status = editCrontab(marker, id, sched, "", reason); } else if (op_flags & OPT_s) { vector sched; if (!(status = getCrontabSched(marker, id, sched))) { cerr << "getCrontabSched failed: " << reason << endl; exit(1); } cout << "sched vec size " << sched.size() << endl; cout << "mins " << sched[0] << " hours " << sched[1] << " days of month " << sched[2] << " months " << sched[3] << " days of week " << sched[4] << endl; exit(0); } else if (op_flags & OPT_c) { if ((status = checkCrontabUnmanaged(marker, cmd))) { cerr << "crontab has unmanaged lines for " << cmd << endl; exit(1); } exit(0); } else { Usage(); } if (!status) { cerr << "editCrontab failed: " << reason << endl; exit(1); } exit(0); } #endif // TEST