Finally HUP-able

This commit is contained in:
Axel Kittenberger 2010-08-04 20:22:38 +00:00
parent e3f7401be4
commit c2354ede16

117
lsyncd.c
View File

@ -386,8 +386,10 @@ struct delay_vector {
* It's not worth to code a dynamic size handling... * It's not worth to code a dynamic size handling...
*/ */
#define MAX_EXCLUDES 256 #define MAX_EXCLUDES 256
char * exclude_dirs[MAX_EXCLUDES] = {NULL, }; struct exclude_vector {
int exclude_dir_n = 0; char * data[MAX_EXCLUDES];
size_t len;
};
/*--------------------------------------------------------------------------* /*--------------------------------------------------------------------------*
* MEMCHECK * MEMCHECK
@ -435,17 +437,30 @@ void maction(const void *nodep, const VISIT which, const int depth) {
*--------------------------------------------------------------------------*/ *--------------------------------------------------------------------------*/
/** /**
* Set to 0 in signal handler, when lsyncd should TERMinate nicely. * Set to 0 in signal handler, when lsyncd should end ASAP.
* This can be either a TERM or a HUP signal.
* In case of HUP killed is 0 and start over.
*/ */
volatile sig_atomic_t keep_going = 1; volatile sig_atomic_t keep_going = 1;
/**
* Received a TERM signal, TERMinate nicely.
*/
volatile sig_atomic_t termed = 0;
/** /**
* Called (out-of-order) when signals arrive * Called (out-of-order) when signals arrive
*/ */
void void
catch_alarm(int sig) catch_alarm(int sig)
{ {
keep_going = 0; switch(sig) {
case SIGTERM :
termed = 1;
/* fall through */
case SIGHUP :
keep_going = 0;
}
} }
/** /**
@ -941,7 +956,7 @@ append_delay(const struct log *log,
struct watch *watch, struct watch *watch,
clock_t alarm) clock_t alarm)
{ {
printlogf(log, DEBUG, "append delay(%s, %d)", watch->dirname, alarm); printlogf(log, DEBUG, "append_delay(%s, %d)", watch->dirname, alarm);
if (watch->delayed) { if (watch->delayed) {
printlogf(log, DEBUG, "ignored since already delayed."); printlogf(log, DEBUG, "ignored since already delayed.");
return false; return false;
@ -1401,6 +1416,7 @@ delay_or_act_dir(const struct global_options *opts,
* @param opts global options * @param opts global options
* @param watches the watch vector * @param watches the watch vector
* @param delays the delay vector * @param delays the delay vector
* @param excludes the excludes vector
* @param inotify_fd inotify file descriptor. * @param inotify_fd inotify file descriptor.
* @param dirname The name or absolute path of the directory to watch. * @param dirname The name or absolute path of the directory to watch.
* @param parent If not NULL, the watches to the parent directory already watched. * @param parent If not NULL, the watches to the parent directory already watched.
@ -1413,6 +1429,7 @@ struct watch *
add_dirwatch(const struct global_options *opts, add_dirwatch(const struct global_options *opts,
struct watch_vector *watches, struct watch_vector *watches,
struct delay_vector *delays, struct delay_vector *delays,
struct exclude_vector *excludes,
int inotify_fd, int inotify_fd,
char const *dirname, char const *dirname,
struct watch *parent, struct watch *parent,
@ -1433,12 +1450,12 @@ add_dirwatch(const struct global_options *opts,
{ {
int i; int i;
for (i = 0; i < exclude_dir_n; i++) { for (i = 0; i < excludes->len; i++) {
if (!strcmp(pathname, exclude_dirs[i])) { if (!strcmp(pathname, excludes->data[i])) {
printlogf(log, NORMAL, "Excluded %s", pathname); printlogf(log, NORMAL, "Excluded %s", pathname);
return NULL; return NULL;
} }
printlogf(log, DEBUG, "comparing %s with %s not an exclude so far.", pathname, exclude_dirs[i]); printlogf(log, DEBUG, "comparing %s with %s not an exclude so far.", pathname, excludes->data[i]);
} }
} }
@ -1488,7 +1505,7 @@ add_dirwatch(const struct global_options *opts,
// add watches if its a directory and not . or .. // add watches if its a directory and not . or ..
if (isdir && strcmp(de->d_name, "..") && strcmp(de->d_name, ".")) { if (isdir && strcmp(de->d_name, "..") && strcmp(de->d_name, ".")) {
// recurse into subdirectories // recurse into subdirectories
struct watch *nw = add_dirwatch(opts, watches, delays, inotify_fd, de->d_name, w, dir_conf); struct watch *nw = add_dirwatch(opts, watches, delays, excludes, inotify_fd, de->d_name, w, dir_conf);
printlogf(log, NORMAL, printlogf(log, NORMAL,
"found new directory: %s in %s -- %s", "found new directory: %s in %s -- %s",
de->d_name, dirname, nw ? "will be synced" : "ignored it"); de->d_name, dirname, nw ? "will be synced" : "ignored it");
@ -1614,6 +1631,7 @@ bool
handle_event(const struct global_options *opts, handle_event(const struct global_options *opts,
struct watch_vector *watches, struct watch_vector *watches,
struct delay_vector *delays, struct delay_vector *delays,
struct exclude_vector *excludes,
int inotify_fd, int inotify_fd,
struct inotify_event *event, struct inotify_event *event,
clock_t alarm) clock_t alarm)
@ -1649,8 +1667,8 @@ handle_event(const struct global_options *opts,
{ {
// TODO, is this needed? or will it be excluded already? // TODO, is this needed? or will it be excluded already?
int i; int i;
for (i = 0; i < exclude_dir_n; i++) { for (i = 0; i < excludes->len; i++) {
if (!strcmp(event->name, exclude_dirs[i])) { if (!strcmp(event->name, excludes->data[i])) {
return true; return true;
} }
} }
@ -1668,7 +1686,7 @@ handle_event(const struct global_options *opts,
if ((IN_ATTRIB | IN_CREATE | IN_CLOSE_WRITE | IN_DELETE | if ((IN_ATTRIB | IN_CREATE | IN_CLOSE_WRITE | IN_DELETE |
IN_MOVED_TO | IN_MOVED_FROM) & event->mask IN_MOVED_TO | IN_MOVED_FROM) & event->mask
) { ) {
printlogf(log, NORMAL, "received event %s:%s.", masktext, event->name); printlogf(log, NORMAL, "received event %s:%s for %d.", masktext, event->name, alarm);
delay_or_act_dir(opts, delays, opts->delay > 0, watch, alarm); delay_or_act_dir(opts, delays, opts->delay > 0, watch, alarm);
} else { } else {
printlogf(log, DEBUG, "... ignored this event."); printlogf(log, DEBUG, "... ignored this event.");
@ -1676,7 +1694,7 @@ handle_event(const struct global_options *opts,
// in case of a new directory create new watches of the subdir // in case of a new directory create new watches of the subdir
if (((IN_CREATE | IN_MOVED_TO) & event->mask) && (IN_ISDIR & event->mask)) { if (((IN_CREATE | IN_MOVED_TO) & event->mask) && (IN_ISDIR & event->mask)) {
add_dirwatch(opts, watches, delays, inotify_fd, event->name, watch, watch->dir_conf); add_dirwatch(opts, watches, delays, excludes, inotify_fd, event->name, watch, watch->dir_conf);
} }
// in case of a removed directory remove watches from the subdir // in case of a removed directory remove watches from the subdir
@ -1693,12 +1711,14 @@ handle_event(const struct global_options *opts,
* @param opts global options * @param opts global options
* @param watches the watch vector * @param watches the watch vector
* @param delays the delay vector * @param delays the delay vector
* @param excludes the exclude vector
* @param inotify_fd inotify file descriptor * @param inotify_fd inotify file descriptor
*/ */
bool bool
master_loop(const struct global_options *opts, master_loop(const struct global_options *opts,
struct watch_vector *watches, struct watch_vector *watches,
struct delay_vector *delays, struct delay_vector *delays,
struct exclude_vector *excludes,
int inotify_fd) int inotify_fd)
{ {
char buf[INOTIFY_BUF_LEN]; char buf[INOTIFY_BUF_LEN];
@ -1780,7 +1800,7 @@ master_loop(const struct global_options *opts,
int i = 0; int i = 0;
while (i < len) { while (i < len) {
struct inotify_event *event = (struct inotify_event *) &buf[i]; struct inotify_event *event = (struct inotify_event *) &buf[i];
handle_event(opts, watches, delays, inotify_fd, event, alarm); handle_event(opts, watches, delays, excludes, inotify_fd, event, alarm);
i += sizeof(struct inotify_event) + event->len; i += sizeof(struct inotify_event) + event->len;
} }
} }
@ -1789,8 +1809,8 @@ master_loop(const struct global_options *opts,
// until one item is found whose expiry time has not yet come // until one item is found whose expiry time has not yet come
// or the stack is empty. Using now time - times(NULL) - everytime // or the stack is empty. Using now time - times(NULL) - everytime
// again as time may progresses while handling delayed entries. // again as time may progresses while handling delayed entries.
while (delays->len > 0 && time_after_eq(times(NULL), delays->data[0])) { while (delays->len > 0 && time_after_eq(times(NULL), delays->data[0]->alarm)) {
printlogf(log, DEBUG, "time for '%s' arrived.", delays->data[0]->dirname); printlogf(log, DEBUG, "time %d for '%s' arrived. now=%d", delays->data[0]->alarm, delays->data[0]->dirname, times(NULL));
rsync_dir(opts, delays->data[0]); rsync_dir(opts, delays->data[0]);
remove_first_delay(delays); remove_first_delay(delays);
} }
@ -2361,9 +2381,13 @@ parse_options(struct global_options *opts, int argc, char **argv)
/** /**
* Parses the exclude file looking for directory masks to not watch. * Parses the exclude file looking for directory masks to not watch.
*
*
*/ */
bool bool
parse_exclude_file(struct log *log, char *filename) { parse_exclude_file(struct log *log,
struct exclude_vector * excludes,
char *filename) {
FILE * ef; FILE * ef;
char line[PATH_MAX+1]; char line[PATH_MAX+1];
int sl; int sl;
@ -2402,7 +2426,7 @@ parse_exclude_file(struct log *log, char *filename) {
} }
if (line[sl - 1] == '/') { if (line[sl - 1] == '/') {
if (exclude_dir_n + 1 >= MAX_EXCLUDES) { if (excludes->len + 1 >= MAX_EXCLUDES) {
printlogf(log, ERROR, printlogf(log, ERROR,
"Too many directory excludes, can only have %d at the most", "Too many directory excludes, can only have %d at the most",
MAX_EXCLUDES); MAX_EXCLUDES);
@ -2419,9 +2443,9 @@ parse_exclude_file(struct log *log, char *filename) {
printlogf(log, NORMAL, "Excluding directories of the name '%s'", line); printlogf(log, NORMAL, "Excluding directories of the name '%s'", line);
exclude_dirs[exclude_dir_n] = s_malloc(log, strlen(line) + 1, "exclude_dir"); excludes->data[excludes->len] = s_malloc(log, strlen(line) + 1, "exclude_dir");
strcpy(exclude_dirs[exclude_dir_n], line); strcpy(excludes->data[excludes->len], line);
exclude_dir_n++; excludes->len++;
} }
} }
@ -2444,16 +2468,18 @@ write_pidfile(const struct log *log, const char *pidfile) {
} }
/** /**
* Main. * Main for one run.
* Can be runned through several times on HUPs.
*/ */
int int
main(int argc, char **argv) one_main(int argc, char **argv)
{ {
struct global_options opts = {{0,}}; // global options struct global_options opts = {{0,}}; // global options
struct log *log = &opts.log; // shortcut to logging options. struct log *log = &opts.log; // shortcut to logging options.
struct watch_vector watches = {0, }; // all watches struct watch_vector watches = {0, }; // all watches
struct delay_vector delays = {0, }; // delayed entries struct delay_vector delays = {0, }; // delayed entries
int inotify_fd; // inotify file descriptor struct exclude_vector excludes = {{0, }}; // excludes
int inotify_fd; // inotify file descriptor
openlog("lsyncd", LOG_CONS | LOG_PID, LOG_DAEMON); openlog("lsyncd", LOG_CONS | LOG_PID, LOG_DAEMON);
@ -2461,7 +2487,7 @@ main(int argc, char **argv)
parse_options(&opts, argc, argv); parse_options(&opts, argc, argv);
if (opts.default_exclude_file) { if (opts.default_exclude_file) {
parse_exclude_file(log, opts.default_exclude_file); parse_exclude_file(log, &excludes, opts.default_exclude_file);
} }
inotify_fd = inotify_init(); inotify_fd = inotify_init();
@ -2499,7 +2525,7 @@ main(int argc, char **argv)
int i; int i;
for (i = 0; i < opts.dir_conf_n; i++) { for (i = 0; i < opts.dir_conf_n; i++) {
printlogf(log, NORMAL, "watching %s", opts.dir_confs[i].source); printlogf(log, NORMAL, "watching %s", opts.dir_confs[i].source);
add_dirwatch(&opts, &watches, &delays, inotify_fd, add_dirwatch(&opts, &watches, &delays, &excludes, inotify_fd,
opts.dir_confs[i].source, NULL, &opts.dir_confs[i]); opts.dir_confs[i].source, NULL, &opts.dir_confs[i]);
} }
} }
@ -2542,9 +2568,13 @@ main(int argc, char **argv)
watches.len); watches.len);
signal(SIGTERM, catch_alarm); signal(SIGTERM, catch_alarm);
signal(SIGHUP, catch_alarm);
master_loop(&opts, &watches, &delays, inotify_fd); master_loop(&opts, &watches, &delays, &excludes, inotify_fd);
if (!termed) {
printlogf(log, NORMAL, "--- Received HUP-Signal, cleaning up and starting over ---");
}
{ {
// memory clean up // memory clean up
int i; int i;
@ -2557,6 +2587,9 @@ main(int argc, char **argv)
} }
s_free(watches.data); s_free(watches.data);
s_free(delays.data); s_free(delays.data);
for(i = 0; i < excludes.len; i++) {
s_free(excludes.data[i]);
}
} }
@ -2568,3 +2601,25 @@ main(int argc, char **argv)
return 0; return 0;
} }
/**
* Main wrapper
*
* Start actual main over and over on HUPs.
*/
int
main(int argc, char **argv)
{
int ret;
do {
ret = one_main(argc, argv);
if (ret) {
return ret;
}
// start over
keep_going = 1;
} while (!termed);
return ret;
}