mirror of
https://github.com/pygos/cron
synced 2024-12-22 01:10:49 +01:00
Initial commit
Signed-off-by: David Oberhollenzer <david.oberhollenzer@tele2.at>
This commit is contained in:
commit
d2b3a983e1
15 changed files with 992 additions and 0 deletions
15
.gitignore
vendored
Normal file
15
.gitignore
vendored
Normal file
|
@ -0,0 +1,15 @@
|
||||||
|
*.o
|
||||||
|
.deps
|
||||||
|
.dirstamp
|
||||||
|
Makefile
|
||||||
|
Makefile.in
|
||||||
|
config.*
|
||||||
|
aclocal.m4
|
||||||
|
autom4te.cache/
|
||||||
|
compile
|
||||||
|
configure
|
||||||
|
depcomp
|
||||||
|
install-sh
|
||||||
|
missing
|
||||||
|
stamp-h1
|
||||||
|
gcrond
|
13
LICENSE
Normal file
13
LICENSE
Normal file
|
@ -0,0 +1,13 @@
|
||||||
|
Copyright (c) 2018 David Oberhollenzer <david.oberhollenzer@tele2.at>
|
||||||
|
|
||||||
|
Permission to use, copy, modify, and distribute this software for any
|
||||||
|
purpose with or without fee is hereby granted, provided that the above
|
||||||
|
copyright notice and this permission notice appear in all copies.
|
||||||
|
|
||||||
|
THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
|
||||||
|
WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
|
||||||
|
MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
|
||||||
|
ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
|
||||||
|
WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
|
||||||
|
ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
|
||||||
|
OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
|
13
Makefile.am
Normal file
13
Makefile.am
Normal file
|
@ -0,0 +1,13 @@
|
||||||
|
ACLOCAL_AMFLAGS = -I m4
|
||||||
|
|
||||||
|
AM_CPPFLAGS = -D_GNU_SOURCE
|
||||||
|
AM_CFLAGS = $(WARN_CFLAGS)
|
||||||
|
|
||||||
|
sbin_PROGRAMS = gcrond
|
||||||
|
|
||||||
|
gcrond_SOURCES = gcrond.c gcrond.h rdcron.c crontab.c cronscan.c rdline.c
|
||||||
|
|
||||||
|
crontabdir = @GCRONDIR@
|
||||||
|
crontab_DATA = crontab/0-example
|
||||||
|
|
||||||
|
EXTRA_DIST = crontab/0-example LICENSE README.md
|
122
README.md
Normal file
122
README.md
Normal file
|
@ -0,0 +1,122 @@
|
||||||
|
# About
|
||||||
|
|
||||||
|
This package contains a small cron implementation called `gcrond`.
|
||||||
|
|
||||||
|
It was written due to a perceived lack of a proper, simple cron
|
||||||
|
implementation. All other cron implementation I came across were either decade
|
||||||
|
old, abandoned pieces of horror ("Cool, I didn't even know that C syntax
|
||||||
|
allows this!") or hopelessly integrated into other, much larger projects (e.g.
|
||||||
|
absorbed by SystemD or in the case of OpenBSD cron, married to special OpenBSD
|
||||||
|
syscalls).
|
||||||
|
|
||||||
|
It was a fun little exercise and it seems to work so far. No idea about
|
||||||
|
standards compliance tough, the implementation was mostly written against
|
||||||
|
the Wikipedia article about Cron.
|
||||||
|
|
||||||
|
## License
|
||||||
|
|
||||||
|
The source code in this package is provided under the OpenBSD flavored ISC
|
||||||
|
license. So you can practically do as you wish, as long as you retain the
|
||||||
|
original copyright notice. The software is provided "as is" (as usual) with
|
||||||
|
no warranty whatsoever (e.g. it might actually do what it was designed for,
|
||||||
|
but it could just as well set your carpet on fire).
|
||||||
|
|
||||||
|
The sub directory `m4` contains third party macro files used by the build
|
||||||
|
system which may be subject to their own, respective licenses.
|
||||||
|
|
||||||
|
|
||||||
|
## Portability
|
||||||
|
|
||||||
|
The program in this package has been written for and tested on a GNU/Linux
|
||||||
|
system, so there may be some GNU-isms in there in addition to Linux specific
|
||||||
|
code. Depending on your target platform, some minor porting effort may be
|
||||||
|
required.
|
||||||
|
|
||||||
|
|
||||||
|
# Building and installing
|
||||||
|
|
||||||
|
This package uses autotools. If you downloaded a distribution tar ball, simply
|
||||||
|
run the `configure` script and then `make` after the Makefile has been
|
||||||
|
generated. A list of possible `configure` options can be viewed by running
|
||||||
|
`configure --help`.
|
||||||
|
|
||||||
|
If you really wish to do so, run `make install` to install the program on your
|
||||||
|
system.
|
||||||
|
|
||||||
|
When working with the git tree, run the `autogen.sh` script to generate the
|
||||||
|
configure script and friends.
|
||||||
|
|
||||||
|
|
||||||
|
# Crontab File Format
|
||||||
|
|
||||||
|
The cron daemon reads its configuration from all files it can find
|
||||||
|
in `/etc/crontab.d/` (exact path can be configured).
|
||||||
|
|
||||||
|
The files are read line by line. Empty lines or lines starting with '#' are
|
||||||
|
skipped.
|
||||||
|
|
||||||
|
Each non-empty line consists of the typical cron fields:
|
||||||
|
|
||||||
|
1. The `minute` field. Legal values are from 0 to 59.
|
||||||
|
2. The `hour` field. Legal values are from 0 to 23.
|
||||||
|
3. The `day of month` field. Legal values are from 1 to 31 (or fewer, depending
|
||||||
|
on the month.
|
||||||
|
4. The `month` field. Legal values are from 1 to 12 (January to December)
|
||||||
|
or the mnemonics `JAN`, `FEB`, `MAR`, `APR`, ...
|
||||||
|
5. The `day of week` field. Legal values are from 0 to 6 (Sunday to Saturday)
|
||||||
|
or the mnemonics `SUN`, `MON`, `TUE`, `WED`, ...
|
||||||
|
6. The command to execute.
|
||||||
|
|
||||||
|
|
||||||
|
The fields are separated by spaces. For the time matching fields, multiple
|
||||||
|
comma separated values can be specified (e.g. `MON,WED,FRI` for a job that
|
||||||
|
should run on Mondays, Wednesdays and Fridays).
|
||||||
|
|
||||||
|
The wild-card character `*` matches any legal value. An stepping can be
|
||||||
|
specified by appending `/` and then a stepping (e.g. for the minute field,
|
||||||
|
`*/5` would let a job run every five minutes).
|
||||||
|
|
||||||
|
A range of values can also be specified as `<lower>-<upper>`, for instance
|
||||||
|
`MON-FRI` would match every day from Monday to Friday (equivalent to `1-5`).
|
||||||
|
|
||||||
|
Intervals and specific values can be combined, for instance a day of month
|
||||||
|
field `*/7,13,25` would trigger once a week, starting from the first of the
|
||||||
|
month (1,7,14,21,28), but additionally include the 13th and the 25th. The
|
||||||
|
same could be expressed as `1-31/7,13,25`.
|
||||||
|
|
||||||
|
|
||||||
|
Instead of specifying a terse cron matching expression, the first five fields
|
||||||
|
can be replaced with one of the following mnemonics:
|
||||||
|
|
||||||
|
- `@yearly` or `@anually` is equivalent to `0 0 1 1 *`, i.e. 1st of January
|
||||||
|
at midnight
|
||||||
|
- `@monthly` is equivalent to `0 0 1 * *`, i.e. 1st of every month at midnight
|
||||||
|
- `@weekly` is equivalent to `0 0 * * 0`, i.e. every Sunday at midnight
|
||||||
|
- `@daily` is equivalent to `0 0 * * *`, i.e. every day at midnight
|
||||||
|
- `@hourly` is equivalent to `0 * * * *`, i.e. every first minute of the hour
|
||||||
|
|
||||||
|
Lastly, the command field is not broken down but passed to `/bin/sh -c`
|
||||||
|
*as is*.
|
||||||
|
|
||||||
|
|
||||||
|
# Security Considerations
|
||||||
|
|
||||||
|
The cron daemon currently has no means of specifying a user to run the jobs as,
|
||||||
|
so if cron runs as root, the jobs it starts do as well. Since by default it
|
||||||
|
reads its configuration from `/etc` which by default is only writable by root,
|
||||||
|
this shouldn't be too much of a problem when using cron for typical system
|
||||||
|
administration tasks.
|
||||||
|
|
||||||
|
If a job should run as another user, tools such as `su`, `runuser`, `setpriv`
|
||||||
|
et cetera need to be used.
|
||||||
|
|
||||||
|
# Possible Future Directions
|
||||||
|
|
||||||
|
The following things would be nice to have:
|
||||||
|
|
||||||
|
- decent logging for cron and the output of the jobs.
|
||||||
|
- cron jobs per user, e.g. scan `~/.crontab.d` or similar and run the collected
|
||||||
|
jobs as the respective user.
|
||||||
|
- timezone handling
|
||||||
|
- some usable strategy for handling time jumps, e.g. caused by a job that
|
||||||
|
syncs time with an NTP server on a system without RTC.
|
3
autogen.sh
Executable file
3
autogen.sh
Executable file
|
@ -0,0 +1,3 @@
|
||||||
|
#!/bin/sh
|
||||||
|
|
||||||
|
autoreconf --force --install --symlink
|
38
configure.ac
Normal file
38
configure.ac
Normal file
|
@ -0,0 +1,38 @@
|
||||||
|
AC_PREREQ([2.60])
|
||||||
|
|
||||||
|
AC_INIT([gcron], [0.1], [david.oberhollenzer@tele2.at], gcron)
|
||||||
|
AC_CONFIG_MACRO_DIR([m4])
|
||||||
|
AM_INIT_AUTOMAKE([foreign subdir-objects dist-xz])
|
||||||
|
AM_SILENT_RULES([yes])
|
||||||
|
AC_PROG_CC
|
||||||
|
AC_PROG_CC_C99
|
||||||
|
AC_PROG_INSTALL
|
||||||
|
AC_PROG_RANLIB
|
||||||
|
|
||||||
|
UL_WARN_ADD([-Wall])
|
||||||
|
UL_WARN_ADD([-Wextra])
|
||||||
|
UL_WARN_ADD([-Wunused])
|
||||||
|
UL_WARN_ADD([-Wmissing-prototypes])
|
||||||
|
UL_WARN_ADD([-Wmissing-declarations])
|
||||||
|
UL_WARN_ADD([-Wwrite-strings])
|
||||||
|
UL_WARN_ADD([-Wjump-misses-init])
|
||||||
|
UL_WARN_ADD([-Wuninitialized])
|
||||||
|
UL_WARN_ADD([-Winit-self])
|
||||||
|
UL_WARN_ADD([-Wlogical-op])
|
||||||
|
UL_WARN_ADD([-Wunused-but-set-parameter])
|
||||||
|
UL_WARN_ADD([-Wunused-but-set-variable])
|
||||||
|
UL_WARN_ADD([-Wunused-parameter])
|
||||||
|
UL_WARN_ADD([-Wunused-result])
|
||||||
|
UL_WARN_ADD([-Wunused-variable])
|
||||||
|
UL_WARN_ADD([-Wduplicated-cond])
|
||||||
|
UL_WARN_ADD([-Wduplicated-branches])
|
||||||
|
UL_WARN_ADD([-Wrestrict])
|
||||||
|
UL_WARN_ADD([-Wnull-dereference])
|
||||||
|
UL_WARN_ADD([-pedantic])
|
||||||
|
|
||||||
|
AC_SUBST([WARN_CFLAGS])
|
||||||
|
|
||||||
|
AC_CONFIG_HEADERS([config.h])
|
||||||
|
AC_DEFINE_DIR(GCRONDIR, sysconfdir/crontab.d, [crontab source directory])
|
||||||
|
|
||||||
|
AC_OUTPUT([Makefile])
|
56
cronscan.c
Normal file
56
cronscan.c
Normal file
|
@ -0,0 +1,56 @@
|
||||||
|
/* SPDX-License-Identifier: ISC */
|
||||||
|
#include "gcrond.h"
|
||||||
|
|
||||||
|
int cronscan(const char *directory, crontab_t **list)
|
||||||
|
{
|
||||||
|
crontab_t *cron, *tail = NULL;
|
||||||
|
struct dirent *ent;
|
||||||
|
int dfd, ret = 0;
|
||||||
|
DIR *dir;
|
||||||
|
|
||||||
|
dir = opendir(directory);
|
||||||
|
if (dir == NULL) {
|
||||||
|
perror(directory);
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
dfd = dirfd(dir);
|
||||||
|
if (dfd < 0) {
|
||||||
|
perror(directory);
|
||||||
|
closedir(dir);
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
for (;;) {
|
||||||
|
errno = 0;
|
||||||
|
ent = readdir(dir);
|
||||||
|
|
||||||
|
if (ent == NULL) {
|
||||||
|
if (errno != 0) {
|
||||||
|
perror(directory);
|
||||||
|
ret = -1;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!strcmp(ent->d_name, ".") || !strcmp(ent->d_name, ".."))
|
||||||
|
continue;
|
||||||
|
|
||||||
|
cron = rdcron(dfd, ent->d_name);
|
||||||
|
if (cron == NULL)
|
||||||
|
continue;
|
||||||
|
|
||||||
|
if (tail == NULL) {
|
||||||
|
*list = cron;
|
||||||
|
tail = cron;
|
||||||
|
} else {
|
||||||
|
tail->next = cron;
|
||||||
|
}
|
||||||
|
|
||||||
|
while (tail->next != NULL)
|
||||||
|
tail = tail->next;
|
||||||
|
}
|
||||||
|
|
||||||
|
closedir(dir);
|
||||||
|
return ret;
|
||||||
|
}
|
61
crontab.c
Normal file
61
crontab.c
Normal file
|
@ -0,0 +1,61 @@
|
||||||
|
/* SPDX-License-Identifier: ISC */
|
||||||
|
#include "gcrond.h"
|
||||||
|
|
||||||
|
void cron_tm_to_mask(crontab_t *out, struct tm *t)
|
||||||
|
{
|
||||||
|
memset(out, 0, sizeof(*out));
|
||||||
|
out->minute = 1UL << ((unsigned long)t->tm_min);
|
||||||
|
out->hour = 1 << t->tm_hour;
|
||||||
|
out->dayofmonth = 1 << (t->tm_mday - 1);
|
||||||
|
out->month = 1 << t->tm_mon;
|
||||||
|
out->dayofweek = 1 << t->tm_wday;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool cron_should_run(const crontab_t *t, const crontab_t *mask)
|
||||||
|
{
|
||||||
|
if ((t->minute & mask->minute) == 0)
|
||||||
|
return false;
|
||||||
|
|
||||||
|
if ((t->hour & mask->hour) == 0)
|
||||||
|
return false;
|
||||||
|
|
||||||
|
if ((t->dayofmonth & mask->dayofmonth) == 0)
|
||||||
|
return false;
|
||||||
|
|
||||||
|
if ((t->month & mask->month) == 0)
|
||||||
|
return false;
|
||||||
|
|
||||||
|
if ((t->dayofweek & mask->dayofweek) == 0)
|
||||||
|
return false;
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
void delcron(crontab_t *cron)
|
||||||
|
{
|
||||||
|
if (cron != NULL) {
|
||||||
|
free(cron->exec);
|
||||||
|
free(cron);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
int runjob(crontab_t *tab)
|
||||||
|
{
|
||||||
|
pid_t pid;
|
||||||
|
|
||||||
|
if (tab->exec == NULL)
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
pid = fork();
|
||||||
|
if (pid == -1) {
|
||||||
|
perror("fork");
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (pid != 0)
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
execl("/bin/sh", "sh", "-c", tab->exec, (char *) 0);
|
||||||
|
perror("runnig shell interpreter");
|
||||||
|
exit(EXIT_FAILURE);
|
||||||
|
}
|
7
crontab/0-example
Normal file
7
crontab/0-example
Normal file
|
@ -0,0 +1,7 @@
|
||||||
|
# +-------- minute (0 - 59)
|
||||||
|
# | +------ hour (0 - 23)
|
||||||
|
# | | +---- day of month (1 - 31)
|
||||||
|
# | | | +-- month (1 - 12)
|
||||||
|
# | | | | +-- day of week (0 - 6)
|
||||||
|
# | | | | |
|
||||||
|
# * * * * * command to execute
|
128
gcrond.c
Normal file
128
gcrond.c
Normal file
|
@ -0,0 +1,128 @@
|
||||||
|
/* SPDX-License-Identifier: ISC */
|
||||||
|
#include "gcrond.h"
|
||||||
|
|
||||||
|
static crontab_t *jobs;
|
||||||
|
static sig_atomic_t run = 1;
|
||||||
|
static sig_atomic_t rescan = 1;
|
||||||
|
|
||||||
|
static void read_config(void)
|
||||||
|
{
|
||||||
|
if (cronscan(GCRONDIR, &jobs)) {
|
||||||
|
fputs("Error reading configuration. Continuing anyway.\n",
|
||||||
|
stderr);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static void cleanup_config(void)
|
||||||
|
{
|
||||||
|
crontab_t *t;
|
||||||
|
|
||||||
|
while (jobs != NULL) {
|
||||||
|
t = jobs;
|
||||||
|
jobs = jobs->next;
|
||||||
|
delcron(t);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static int timeout_minutes(int minutes)
|
||||||
|
{
|
||||||
|
time_t now = time(NULL);
|
||||||
|
struct tm t;
|
||||||
|
|
||||||
|
localtime_r(&now, &t);
|
||||||
|
return minutes * 60 + 30 - t.tm_sec;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int calc_timeout(void)
|
||||||
|
{
|
||||||
|
time_t now = time(NULL), future;
|
||||||
|
struct tm tmstruct;
|
||||||
|
crontab_t mask, *t;
|
||||||
|
int minutes;
|
||||||
|
|
||||||
|
for (minutes = 0; minutes < 120; ++minutes) {
|
||||||
|
future = now + minutes * 60;
|
||||||
|
|
||||||
|
localtime_r(&future, &tmstruct);
|
||||||
|
cron_tm_to_mask(&mask, &tmstruct);
|
||||||
|
|
||||||
|
for (t = jobs; t != NULL; t = t->next) {
|
||||||
|
if (cron_should_run(t, &mask))
|
||||||
|
goto out;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
out:
|
||||||
|
return timeout_minutes(minutes ? minutes : 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void runjobs(void)
|
||||||
|
{
|
||||||
|
time_t now = time(NULL);
|
||||||
|
struct tm tmstruct;
|
||||||
|
crontab_t mask, *t;
|
||||||
|
|
||||||
|
localtime_r(&now, &tmstruct);
|
||||||
|
cron_tm_to_mask(&mask, &tmstruct);
|
||||||
|
|
||||||
|
for (t = jobs; t != NULL; t = t->next) {
|
||||||
|
if (cron_should_run(t, &mask))
|
||||||
|
runjob(t);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static void sighandler(int signo)
|
||||||
|
{
|
||||||
|
pid_t pid;
|
||||||
|
|
||||||
|
switch (signo) {
|
||||||
|
case SIGINT:
|
||||||
|
case SIGTERM:
|
||||||
|
run = 0;
|
||||||
|
break;
|
||||||
|
case SIGHUP:
|
||||||
|
rescan = 1;
|
||||||
|
break;
|
||||||
|
case SIGCHLD:
|
||||||
|
while ((pid = waitpid(-1, NULL, WNOHANG)) != -1)
|
||||||
|
;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
int main(void)
|
||||||
|
{
|
||||||
|
struct timespec stime;
|
||||||
|
struct sigaction act;
|
||||||
|
int timeout;
|
||||||
|
|
||||||
|
memset(&act, 0, sizeof(act));
|
||||||
|
act.sa_handler = sighandler;
|
||||||
|
sigaction(SIGINT, &act, NULL);
|
||||||
|
sigaction(SIGTERM, &act, NULL);
|
||||||
|
sigaction(SIGHUP, &act, NULL);
|
||||||
|
sigaction(SIGCHLD, &act, NULL);
|
||||||
|
|
||||||
|
while (run) {
|
||||||
|
if (rescan == 1) {
|
||||||
|
cleanup_config();
|
||||||
|
read_config();
|
||||||
|
timeout = timeout_minutes(1);
|
||||||
|
rescan = 0;
|
||||||
|
} else {
|
||||||
|
runjobs();
|
||||||
|
timeout = calc_timeout();
|
||||||
|
}
|
||||||
|
|
||||||
|
stime.tv_sec = timeout;
|
||||||
|
stime.tv_nsec = 0;
|
||||||
|
|
||||||
|
while (nanosleep(&stime, &stime) != 0 && run && !rescan) {
|
||||||
|
if (errno != EINTR) {
|
||||||
|
perror("nanosleep");
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return EXIT_SUCCESS;
|
||||||
|
}
|
66
gcrond.h
Normal file
66
gcrond.h
Normal file
|
@ -0,0 +1,66 @@
|
||||||
|
/* SPDX-License-Identifier: ISC */
|
||||||
|
#ifndef GCROND_H
|
||||||
|
#define GCROND_H
|
||||||
|
|
||||||
|
#include <sys/types.h>
|
||||||
|
#include <sys/stat.h>
|
||||||
|
#include <sys/wait.h>
|
||||||
|
#include <stdbool.h>
|
||||||
|
#include <stddef.h>
|
||||||
|
#include <string.h>
|
||||||
|
#include <stdlib.h>
|
||||||
|
#include <stdint.h>
|
||||||
|
#include <unistd.h>
|
||||||
|
#include <limits.h>
|
||||||
|
#include <signal.h>
|
||||||
|
#include <dirent.h>
|
||||||
|
#include <stdarg.h>
|
||||||
|
#include <ctype.h>
|
||||||
|
#include <stdio.h>
|
||||||
|
#include <errno.h>
|
||||||
|
#include <fcntl.h>
|
||||||
|
#include <time.h>
|
||||||
|
|
||||||
|
#include "config.h"
|
||||||
|
|
||||||
|
#define ARRAY_SIZE(x) (sizeof(x) / sizeof((x)[0]))
|
||||||
|
|
||||||
|
typedef struct crontab_t {
|
||||||
|
struct crontab_t *next;
|
||||||
|
char *exec;
|
||||||
|
|
||||||
|
uint64_t minute;
|
||||||
|
uint32_t hour;
|
||||||
|
uint32_t dayofmonth;
|
||||||
|
uint16_t month;
|
||||||
|
uint8_t dayofweek;
|
||||||
|
} crontab_t;
|
||||||
|
|
||||||
|
typedef struct {
|
||||||
|
const char *filename; /* input file name */
|
||||||
|
size_t lineno; /* current line number */
|
||||||
|
FILE *fp;
|
||||||
|
char *line;
|
||||||
|
} rdline_t;
|
||||||
|
|
||||||
|
int rdline_init(rdline_t *t, int dirfd, const char *filename);
|
||||||
|
|
||||||
|
void rdline_complain(rdline_t *t, const char *msg, ...);
|
||||||
|
|
||||||
|
void rdline_cleanup(rdline_t *t);
|
||||||
|
|
||||||
|
int rdline(rdline_t *t);
|
||||||
|
|
||||||
|
crontab_t *rdcron(int dirfd, const char *filename);
|
||||||
|
|
||||||
|
void delcron(crontab_t *cron);
|
||||||
|
|
||||||
|
int cronscan(const char *directory, crontab_t **list);
|
||||||
|
|
||||||
|
void cron_tm_to_mask(crontab_t *out, struct tm *t);
|
||||||
|
|
||||||
|
bool cron_should_run(const crontab_t *t, const crontab_t *mask);
|
||||||
|
|
||||||
|
int runjob(crontab_t *tab);
|
||||||
|
|
||||||
|
#endif /* GCROND_H */
|
35
m4/ac_define_dir.m4
Normal file
35
m4/ac_define_dir.m4
Normal file
|
@ -0,0 +1,35 @@
|
||||||
|
dnl @synopsis AC_DEFINE_DIR(VARNAME, DIR [, DESCRIPTION])
|
||||||
|
dnl
|
||||||
|
dnl This macro sets VARNAME to the expansion of the DIR variable,
|
||||||
|
dnl taking care of fixing up ${prefix} and such.
|
||||||
|
dnl
|
||||||
|
dnl VARNAME is then offered as both an output variable and a C
|
||||||
|
dnl preprocessor symbol.
|
||||||
|
dnl
|
||||||
|
dnl Example:
|
||||||
|
dnl
|
||||||
|
dnl AC_DEFINE_DIR([DATADIR], [datadir], [Where data are placed to.])
|
||||||
|
dnl
|
||||||
|
dnl @category Misc
|
||||||
|
dnl @author Stepan Kasal <kasal@ucw.cz>
|
||||||
|
dnl @author Andreas Schwab <schwab@suse.de>
|
||||||
|
dnl @author Guido U. Draheim <guidod@gmx.de>
|
||||||
|
dnl @author Alexandre Oliva
|
||||||
|
dnl @version 2006-10-13
|
||||||
|
dnl @license AllPermissive
|
||||||
|
|
||||||
|
AC_DEFUN([AC_DEFINE_DIR], [
|
||||||
|
prefix_NONE=
|
||||||
|
exec_prefix_NONE=
|
||||||
|
test "x$prefix" = xNONE && prefix_NONE=yes && prefix=$ac_default_prefix
|
||||||
|
test "x$exec_prefix" = xNONE && exec_prefix_NONE=yes && exec_prefix=$prefix
|
||||||
|
dnl In Autoconf 2.60, ${datadir} refers to ${datarootdir}, which in turn
|
||||||
|
dnl refers to ${prefix}. Thus we have to use `eval' twice.
|
||||||
|
eval ac_define_dir="\"[$]$2\""
|
||||||
|
eval ac_define_dir="\"$ac_define_dir\""
|
||||||
|
AC_SUBST($1, "$ac_define_dir")
|
||||||
|
AC_DEFINE_UNQUOTED($1, "$ac_define_dir", [$3])
|
||||||
|
test "$prefix_NONE" && prefix=NONE
|
||||||
|
test "$exec_prefix_NONE" && exec_prefix=NONE
|
||||||
|
])
|
||||||
|
|
40
m4/compiler.m4
Normal file
40
m4/compiler.m4
Normal file
|
@ -0,0 +1,40 @@
|
||||||
|
dnl Copyright (C) 2008-2011 Free Software Foundation, Inc.
|
||||||
|
dnl This file is free software; the Free Software Foundation
|
||||||
|
dnl gives unlimited permission to copy and/or distribute it,
|
||||||
|
dnl with or without modifications, as long as this notice is preserved.
|
||||||
|
|
||||||
|
dnl From Simon Josefsson
|
||||||
|
dnl -- derivated from coreutils m4/warnings.m4
|
||||||
|
|
||||||
|
# UL_AS_VAR_APPEND(VAR, VALUE)
|
||||||
|
# ----------------------------
|
||||||
|
# Provide the functionality of AS_VAR_APPEND if Autoconf does not have it.
|
||||||
|
m4_ifdef([AS_VAR_APPEND],
|
||||||
|
[m4_copy([AS_VAR_APPEND], [UL_AS_VAR_APPEND])],
|
||||||
|
[m4_define([UL_AS_VAR_APPEND],
|
||||||
|
[AS_VAR_SET([$1], [AS_VAR_GET([$1])$2])])])
|
||||||
|
|
||||||
|
# UL_ADD_WARN(COMPILER_OPTION [, VARNAME])
|
||||||
|
# ------------------------
|
||||||
|
# Adds parameter to WARN_CFLAGS (or to $VARNAME) if the compiler supports it.
|
||||||
|
AC_DEFUN([UL_WARN_ADD], [
|
||||||
|
m4_define([warnvarname], m4_default([$2],WARN_CFLAGS))
|
||||||
|
AS_VAR_PUSHDEF([ul_Warn], [ul_cv_warn_$1])dnl
|
||||||
|
AC_CACHE_CHECK([whether compiler handles $1], m4_defn([ul_Warn]), [
|
||||||
|
# store AC_LANG_WERROR status, then turn it on
|
||||||
|
save_ac_[]_AC_LANG_ABBREV[]_werror_flag="${ac_[]_AC_LANG_ABBREV[]_werror_flag}"
|
||||||
|
AC_LANG_WERROR
|
||||||
|
|
||||||
|
ul_save_CPPFLAGS="$CPPFLAGS"
|
||||||
|
CPPFLAGS="-Werror ${CPPFLAGS} $1"
|
||||||
|
AC_PREPROC_IFELSE([AC_LANG_PROGRAM([])],
|
||||||
|
[AS_VAR_SET(ul_Warn, [yes])],
|
||||||
|
[AS_VAR_SET(ul_Warn, [no])])
|
||||||
|
# restore AC_LANG_WERROR
|
||||||
|
ac_[]_AC_LANG_ABBREV[]_werror_flag="${save_ac_[]_AC_LANG_ABBREV[]_werror_flag}"
|
||||||
|
|
||||||
|
CPPFLAGS="$ul_save_CPPFLAGS"
|
||||||
|
])
|
||||||
|
AS_VAR_IF(ul_Warn, [yes], [UL_AS_VAR_APPEND(warnvarname, [" $1"])])
|
||||||
|
])
|
||||||
|
|
318
rdcron.c
Normal file
318
rdcron.c
Normal file
|
@ -0,0 +1,318 @@
|
||||||
|
/* SPDX-License-Identifier: ISC */
|
||||||
|
#include "gcrond.h"
|
||||||
|
|
||||||
|
typedef struct {
|
||||||
|
const char *name;
|
||||||
|
int value;
|
||||||
|
} enum_map_t;
|
||||||
|
|
||||||
|
static const enum_map_t weekday[] = {
|
||||||
|
{ "MON", 1 },
|
||||||
|
{ "TUE", 2 },
|
||||||
|
{ "WED", 3 },
|
||||||
|
{ "THU", 4 },
|
||||||
|
{ "FRI", 5 },
|
||||||
|
{ "SAT", 6 },
|
||||||
|
{ "SUN", 0 },
|
||||||
|
{ NULL, 0 },
|
||||||
|
};
|
||||||
|
|
||||||
|
static const enum_map_t month[] = {
|
||||||
|
{ "JAN", 1 },
|
||||||
|
{ "FEB", 2 },
|
||||||
|
{ "MAR", 3 },
|
||||||
|
{ "APR", 4 },
|
||||||
|
{ "MAY", 5 },
|
||||||
|
{ "JUN", 6 },
|
||||||
|
{ "JUL", 7 },
|
||||||
|
{ "AUG", 8 },
|
||||||
|
{ "SEP", 9 },
|
||||||
|
{ "OCT", 10 },
|
||||||
|
{ "NOV", 11 },
|
||||||
|
{ "DEC", 12 },
|
||||||
|
{ NULL, 0 },
|
||||||
|
};
|
||||||
|
|
||||||
|
static const struct {
|
||||||
|
const char *macro;
|
||||||
|
crontab_t tab;
|
||||||
|
} intervals[] = {
|
||||||
|
{
|
||||||
|
.macro = "yearly",
|
||||||
|
.tab = {
|
||||||
|
.minute = 0x01,
|
||||||
|
.hour = 0x01,
|
||||||
|
.dayofmonth = 0x01,
|
||||||
|
.month = 0x01,
|
||||||
|
.dayofweek = 0xFF
|
||||||
|
},
|
||||||
|
}, {
|
||||||
|
.macro = "annually",
|
||||||
|
.tab = {
|
||||||
|
.minute = 0x01,
|
||||||
|
.hour = 0x01,
|
||||||
|
.dayofmonth = 0x01,
|
||||||
|
.month = 0x01,
|
||||||
|
.dayofweek = 0xFF
|
||||||
|
},
|
||||||
|
}, {
|
||||||
|
.macro = "monthly",
|
||||||
|
.tab = {
|
||||||
|
.minute = 0x01,
|
||||||
|
.hour = 0x01,
|
||||||
|
.dayofmonth = 0x01,
|
||||||
|
.month = 0xFFFF,
|
||||||
|
.dayofweek = 0xFF
|
||||||
|
},
|
||||||
|
}, {
|
||||||
|
.macro = "weekly",
|
||||||
|
.tab = {
|
||||||
|
.minute = 0x01,
|
||||||
|
.hour = 0x01,
|
||||||
|
.dayofmonth = 0xFFFFFFFF,
|
||||||
|
.month = 0xFFFF,
|
||||||
|
.dayofweek = 0x01
|
||||||
|
},
|
||||||
|
}, {
|
||||||
|
.macro = "daily",
|
||||||
|
.tab = {
|
||||||
|
.minute = 0x01,
|
||||||
|
.hour = 0x01,
|
||||||
|
.dayofmonth = 0xFFFFFFFF,
|
||||||
|
.month = 0xFFFF,
|
||||||
|
.dayofweek = 0xFF
|
||||||
|
},
|
||||||
|
}, {
|
||||||
|
.macro = "hourly",
|
||||||
|
.tab = {
|
||||||
|
.minute = 0x01,
|
||||||
|
.hour = 0xFFFFFFFF,
|
||||||
|
.dayofmonth = 0xFFFFFFFF,
|
||||||
|
.month = 0xFFFF,
|
||||||
|
.dayofweek = 0xFF
|
||||||
|
},
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
/*****************************************************************************/
|
||||||
|
|
||||||
|
static char *readnum(char *line, int *out, int minval, int maxval,
|
||||||
|
const enum_map_t *mnemonic, rdline_t *rd)
|
||||||
|
{
|
||||||
|
int i, temp, value = 0;
|
||||||
|
const enum_map_t *ev;
|
||||||
|
|
||||||
|
if (!isdigit(*line)) {
|
||||||
|
if (!mnemonic)
|
||||||
|
goto fail_mn;
|
||||||
|
|
||||||
|
for (i = 0; isalnum(line[i]); ++i)
|
||||||
|
;
|
||||||
|
if (i == 0)
|
||||||
|
goto fail_mn;
|
||||||
|
|
||||||
|
temp = line[i];
|
||||||
|
line[i] = '\0';
|
||||||
|
|
||||||
|
for (ev = mnemonic; ev->name != NULL; ++ev) {
|
||||||
|
if (!strcmp(line, mnemonic->name) == 0)
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (ev->name == NULL) {
|
||||||
|
rdline_complain(rd, "unexpected '%s'", line);
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
line[i] = temp;
|
||||||
|
*out = ev->value;
|
||||||
|
return line + i;
|
||||||
|
}
|
||||||
|
|
||||||
|
while (isdigit(*line)) {
|
||||||
|
i = ((*(line++)) - '0');
|
||||||
|
if (value > (maxval - i) / 10)
|
||||||
|
goto fail_of;
|
||||||
|
value = value * 10 + i;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (value < minval)
|
||||||
|
goto fail_uf;
|
||||||
|
|
||||||
|
*out = value;
|
||||||
|
return line;
|
||||||
|
fail_of:
|
||||||
|
rdline_complain(rd, "value exceeds maximum (%d > %d)", value, maxval);
|
||||||
|
return NULL;
|
||||||
|
fail_uf:
|
||||||
|
rdline_complain(rd, "value too small (%d < %d)", value, minval);
|
||||||
|
return NULL;
|
||||||
|
fail_mn:
|
||||||
|
rdline_complain(rd, "expected numeric value");
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
static char *readfield(char *line, uint64_t *out, int minval, int maxval,
|
||||||
|
const enum_map_t *mnemonic, rdline_t *rd)
|
||||||
|
{
|
||||||
|
int value, endvalue, step;
|
||||||
|
uint64_t v = 0;
|
||||||
|
next:
|
||||||
|
if (*line == '*') {
|
||||||
|
++line;
|
||||||
|
value = minval;
|
||||||
|
endvalue = maxval;
|
||||||
|
} else {
|
||||||
|
line = readnum(line, &value, minval, maxval, mnemonic, rd);
|
||||||
|
if (!line)
|
||||||
|
goto fail;
|
||||||
|
|
||||||
|
if (*line == '-') {
|
||||||
|
line = readnum(line + 1, &endvalue, minval, maxval,
|
||||||
|
mnemonic, rd);
|
||||||
|
if (!line)
|
||||||
|
goto fail;
|
||||||
|
} else {
|
||||||
|
endvalue = value;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (endvalue < value)
|
||||||
|
goto fail;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (*line == '/') {
|
||||||
|
line = readnum(line + 1, &step, 1, maxval + 1, NULL, rd);
|
||||||
|
if (!line)
|
||||||
|
goto fail;
|
||||||
|
} else {
|
||||||
|
step = 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
while (value <= endvalue) {
|
||||||
|
v |= 1UL << (unsigned long)(value - minval);
|
||||||
|
value += step;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (*line == ',') {
|
||||||
|
++line;
|
||||||
|
goto next;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (*line != '\0' && !isspace(*line))
|
||||||
|
goto fail;
|
||||||
|
while (isspace(*line))
|
||||||
|
++line;
|
||||||
|
|
||||||
|
*out = v;
|
||||||
|
return line;
|
||||||
|
fail:
|
||||||
|
rdline_complain(rd, "invalid time range expression");
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*****************************************************************************/
|
||||||
|
|
||||||
|
static char *cron_interval(crontab_t *cron, rdline_t *rd)
|
||||||
|
{
|
||||||
|
char *arg = rd->line;
|
||||||
|
size_t i, j;
|
||||||
|
|
||||||
|
if (*(arg++) != '@')
|
||||||
|
goto fail;
|
||||||
|
for (j = 0; isalpha(arg[j]); ++j)
|
||||||
|
;
|
||||||
|
if (j == 0 || !isspace(arg[j]))
|
||||||
|
goto fail;
|
||||||
|
|
||||||
|
for (i = 0; i < ARRAY_SIZE(intervals); ++i) {
|
||||||
|
if (strlen(intervals[i].macro) != j)
|
||||||
|
continue;
|
||||||
|
if (strncmp(intervals[i].macro, arg, j) == 0)
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (i == ARRAY_SIZE(intervals))
|
||||||
|
goto fail;
|
||||||
|
|
||||||
|
cron->minute = intervals[i].tab.minute;
|
||||||
|
cron->hour = intervals[i].tab.hour;
|
||||||
|
cron->dayofmonth = intervals[i].tab.dayofmonth;
|
||||||
|
cron->month = intervals[i].tab.month;
|
||||||
|
cron->dayofweek = intervals[i].tab.dayofweek;
|
||||||
|
return arg + j;
|
||||||
|
fail:
|
||||||
|
rdline_complain(rd, "unknown interval '%s'", arg);
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
static char *cron_fields(crontab_t *cron, rdline_t *rd)
|
||||||
|
{
|
||||||
|
char *arg = rd->line;
|
||||||
|
uint64_t value;
|
||||||
|
|
||||||
|
if ((arg = readfield(arg, &value, 0, 59, NULL, rd)) == NULL)
|
||||||
|
return NULL;
|
||||||
|
cron->minute = value;
|
||||||
|
|
||||||
|
if ((arg = readfield(arg, &value, 0, 23, NULL, rd)) == NULL)
|
||||||
|
return NULL;
|
||||||
|
cron->hour = value;
|
||||||
|
|
||||||
|
if ((arg = readfield(arg, &value, 1, 31, NULL, rd)) == NULL)
|
||||||
|
return NULL;
|
||||||
|
cron->dayofmonth = value;
|
||||||
|
|
||||||
|
if ((arg = readfield(arg, &value, 1, 12, month, rd)) == NULL)
|
||||||
|
return NULL;
|
||||||
|
cron->month = value;
|
||||||
|
|
||||||
|
if ((arg = readfield(arg, &value, 0, 6, weekday, rd)) == NULL)
|
||||||
|
return NULL;
|
||||||
|
cron->dayofweek = value;
|
||||||
|
|
||||||
|
return arg;
|
||||||
|
}
|
||||||
|
|
||||||
|
crontab_t *rdcron(int dirfd, const char *filename)
|
||||||
|
{
|
||||||
|
crontab_t *cron, *list = NULL;
|
||||||
|
rdline_t rd;
|
||||||
|
char *ptr;
|
||||||
|
|
||||||
|
if (rdline_init(&rd, dirfd, filename))
|
||||||
|
return NULL;
|
||||||
|
|
||||||
|
while (rdline(&rd) == 0) {
|
||||||
|
cron = calloc(1, sizeof(*cron));
|
||||||
|
if (cron == NULL) {
|
||||||
|
rdline_complain(&rd, strerror(errno));
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (rd.line[0] == '@') {
|
||||||
|
ptr = cron_interval(cron, &rd);
|
||||||
|
} else {
|
||||||
|
ptr = cron_fields(cron, &rd);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (ptr == NULL) {
|
||||||
|
free(cron);
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
while (isspace(*ptr))
|
||||||
|
++ptr;
|
||||||
|
|
||||||
|
cron->exec = strdup(ptr);
|
||||||
|
if (cron->exec == NULL) {
|
||||||
|
rdline_complain(&rd, strerror(errno));
|
||||||
|
free(cron);
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
cron->next = list;
|
||||||
|
list = cron;
|
||||||
|
}
|
||||||
|
|
||||||
|
rdline_cleanup(&rd);
|
||||||
|
return list;
|
||||||
|
}
|
77
rdline.c
Normal file
77
rdline.c
Normal file
|
@ -0,0 +1,77 @@
|
||||||
|
/* SPDX-License-Identifier: ISC */
|
||||||
|
#include "gcrond.h"
|
||||||
|
|
||||||
|
int rdline(rdline_t *t)
|
||||||
|
{
|
||||||
|
size_t i, len;
|
||||||
|
|
||||||
|
do {
|
||||||
|
free(t->line);
|
||||||
|
t->line = NULL;
|
||||||
|
errno = 0;
|
||||||
|
len = 0;
|
||||||
|
|
||||||
|
if (getline(&t->line, &len, t->fp) < 0) {
|
||||||
|
if (errno) {
|
||||||
|
rdline_complain(t, strerror(errno));
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
t->lineno += 1;
|
||||||
|
|
||||||
|
for (i = 0; isspace(t->line[i]); ++i)
|
||||||
|
;
|
||||||
|
|
||||||
|
if (t->line[i] == '\0' || t->line[i] == '#') {
|
||||||
|
t->line[0] = '\0';
|
||||||
|
} else if (i) {
|
||||||
|
memmove(t->line, t->line + i, len - i + 1);
|
||||||
|
}
|
||||||
|
} while (t->line[0] == '\0');
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
void rdline_complain(rdline_t *t, const char *msg, ...)
|
||||||
|
{
|
||||||
|
va_list ap;
|
||||||
|
|
||||||
|
fprintf(stderr, "%s: %zu: ", t->filename, t->lineno);
|
||||||
|
|
||||||
|
va_start(ap, msg);
|
||||||
|
vfprintf(stderr, msg, ap);
|
||||||
|
va_end(ap);
|
||||||
|
|
||||||
|
fputc('\n', stderr);
|
||||||
|
}
|
||||||
|
|
||||||
|
int rdline_init(rdline_t *t, int dirfd, const char *filename)
|
||||||
|
{
|
||||||
|
int fd;
|
||||||
|
|
||||||
|
memset(t, 0, sizeof(*t));
|
||||||
|
|
||||||
|
fd = openat(dirfd, filename, O_RDONLY);
|
||||||
|
if (fd == -1) {
|
||||||
|
perror(filename);
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
t->fp = fdopen(fd, "r");
|
||||||
|
if (t->fp == NULL) {
|
||||||
|
perror("fdopen");
|
||||||
|
close(fd);
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
t->filename = filename;
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
void rdline_cleanup(rdline_t *t)
|
||||||
|
{
|
||||||
|
free(t->line);
|
||||||
|
fclose(t->fp);
|
||||||
|
}
|
Loading…
Reference in a new issue