/** -*- tab-width: 2; -*- * lsyncd.c Live (Mirror) Syncing Demon * * License: GPLv2 (see COPYING) or any later version * * Authors: Axel Kittenberger * Eugene Sanivsky */ #include "config.h" #define _GNU_SOURCE #include #include #include #ifdef HAVE_SYS_INOTIFY_H # include #else # include "inotify-nosys.h" #endif #include #include #include #include #include #include #include #include #include #include #include #include #ifdef XML_CONFIG #include #include #endif #define INOTIFY_BUF_LEN (512 * (sizeof(struct inotify_event) + 16)) #define LOG_DEBUG 1 #define LOG_NORMAL 2 #define LOG_ERROR 3 /** * Possible Exit codes for this application */ enum lsyncd_exit_code { LSYNCD_SUCCESS = 0, /* out-of memory */ LSYNCD_OUTOFMEMORY = 1, /* file was not found, or failed to write */ LSYNCD_FILENOTFOUND = 2, /* execution somehow failed */ LSYNCD_EXECFAIL = 3, /* command-line arguments given to lsyncd are bad */ LSYNCD_BADPARAMETERS = 4, /* Too many excludes files were specified */ LSYNCD_TOOMANYDIRECTORYEXCLUDES = 5, /* something wrong with the config file */ LSYNCD_BADCONFIGFILE = 6, /* Internal exit code to denote that execv failed */ LSYNCD_INTERNALFAIL = 255, }; /** * An option paramater for the action call can either be: */ enum call_option_kind { CO_EOL, // end of list, CO_TEXT, // specified by text, CO_EXCLUDE, // be the exclude file CO_SOURCE, // be the source of the operation CO_DEST, // be the destination of the operation }; /*--------------------------------------------------------------------------* * Structure definitions *--------------------------------------------------------------------------*/ /** * An option parameter for the call. */ struct call_option { /** * The kind of this option. */ enum call_option_kind kind; /** * The text if its text. */ char *text; }; /** * A configurated directory to sync (including subdirectories) * * In case of beeing called with simple argument without a config file * there will be exactly one sync_directory. */ struct dir_conf { /** * Source dir to watch (no default value) */ char * source; /** * NULL terminated list of targets to rsync to (no default value). */ char ** targets; /** * binary to call (defaults to global default setting) */ char * binary; /** * the options to call the binary (defaults to global default setting) */ struct call_option * callopts; /** * the exclude-file to pass to rsync (defaults to global default setting) * TODO, Currently ignored! */ char * exclude_file; }; /** * Structure to store the directory watches of the deamon. */ struct dir_watch { /** * The watch descriptor returned by kernel. */ int wd; /** * The name of the directory. * In case of the root dir to be watched, it is a full path * and parent == NULL. Otherwise its just the name of the * directory and parent points to the parent directory thats * also watched. */ char * dirname; /** * Call this directory that way on destiation. * if NULL call it like dirname. */ char * destname; /** * Points to the index of the parent. * -1 if no parent */ int parent; /** * The applicable configuration for this directory */ struct dir_conf * dir_conf; }; /** * Structure to store strings for the diversve inotfy masked events. * Actually used for comfortable debugging only. */ struct inotify_mask_text { int mask; char const * text; }; /*--------------------------------------------------------------------------* * Global variables *--------------------------------------------------------------------------*/ /** * Global Option: The loglevel is how eloquent lsyncd will be. */ int loglevel = LOG_NORMAL; /** * Global Option: if true no action will actually be called. */ int flag_dryrun = 0; /** * Global Option: if true, do not detach and log to stdout/stderr. */ int flag_nodaemon = 0; /** * Global Option: pidfile, which holds the PID of the running daemon process. */ char * pidfile = NULL; #ifdef XML_CONFIG /** * Global Option: the filename to read config from. */ char * conf_filename = "/etc/lsyncd.conf.xml"; #endif /** * Global Option: this binary is used if no other specified in dir_conf. */ char * default_binary = "/usr/bin/rsync"; /** * Global Option: default exclude file */ char * default_exclude_file = NULL; /** * Standard default options to call the binary with. */ struct call_option standard_callopts[] = { { CO_TEXT, "-lt%r" }, { CO_TEXT, "--delete" }, { CO_EXCLUDE, NULL }, { CO_SOURCE, NULL }, { CO_DEST, NULL }, { CO_EOL, NULL }, }; /** * Global Option: default options to call the binary with. */ struct call_option * default_callopts = standard_callopts; /** * The configuratiton for dirs to synchronize */ struct dir_conf * dir_confs = NULL; /** * The number of configurated dirs to sync. */ int dir_conf_n = 0; /** * A stack of offset pointers to dir_watches to directories to sync. */ int *tosync = NULL; /** * Number of ints allocaetd for tosync stack */ int tosync_size = 0; /** * The pointer of the current tosync position. */ int tosync_pos = 0; /** * A constant that assigns every inotify mask a printable string. * Used for debugging. */ struct inotify_mask_text mask_texts[] = { { IN_ACCESS, "ACCESS" }, { IN_ATTRIB, "ATTRIB" }, { IN_CLOSE_WRITE, "CLOSE_WRITE" }, { IN_CLOSE_NOWRITE, "CLOSE_NOWRITE" }, { IN_CREATE, "CREATE" }, { IN_DELETE, "DELETE" }, { IN_DELETE_SELF, "DELETE_SELF" }, { IN_IGNORED, "IGNORED" }, { IN_MODIFY, "MODIFY" }, { IN_MOVE_SELF, "MOVE_SELF" }, { IN_MOVED_FROM, "MOVED_FROM" }, { IN_MOVED_TO, "MOVED_TO" }, { IN_OPEN, "OPEN" }, { 0, "" }, }; /** * Holds an allocated array of all directories watched. */ struct dir_watch *dir_watches; /** * The allocated size of dir_watches; */ int dir_watch_size = 0; /** * The largest dir_watch number used; */ int dir_watch_num = 0; /** * lsyncd will log into this file/stream. */ char * logfile = "/var/log/lsyncd"; /** * The inotify instance. */ int inotf; /** * Array of strings of directory names to include. * This is limited to MAX_EXCLUDES. * It's not worth to code a dynamic size handling... */ #define MAX_EXCLUDES 256 char * exclude_dirs[MAX_EXCLUDES] = {NULL, }; int exclude_dir_n = 0; /*--------------------------------------------------------------------------* * Small generic helper routines. * (signal catching, memory fetching, message output) *--------------------------------------------------------------------------*/ /** * Set to 0 in signal handler, when lsyncd should TERMinate nicely. */ volatile sig_atomic_t keep_going = 1; /** * Called (out-of-order) when signals arrive */ void catch_alarm(int sig) { keep_going = 0; } /** * Prints a message to the log stream, preceding a timestamp. * Otherwise it behaves like printf(); */ void printlogf(int level, const char *fmt, ...) { va_list ap; char * ct; time_t mtime; FILE * flog; if (level < loglevel) { return; } if (!flag_nodaemon) { flog = fopen(logfile, "a"); if (flog == NULL) { fprintf(stderr, "cannot open logfile [%s]!\n", logfile); exit(LSYNCD_FILENOTFOUND); } } else { flog = stdout; } va_start(ap, fmt); time(&mtime); ct = ctime(&mtime); ct[strlen(ct) - 1] = 0; // cut trailing \n fprintf(flog, "%s: ", ct); switch (level) { case LOG_DEBUG : break; case LOG_NORMAL : break; case LOG_ERROR : fprintf(flog, "ERROR: "); break; } vfprintf(flog, fmt, ap); fprintf(flog, "\n"); va_end(ap); if (!flag_nodaemon) { fclose(flog); } } /** * "secured" malloc, meaning the deamon shall kill itself * in case of out of memory. * * On linux systems, which is actually the only system this * deamon will run at, due to the use of inotify, this is * an "academic" cleaness only, linux will never return out * memory, but kill a process to ensure memory will be * available. */ void *s_malloc(size_t size) { void *r = malloc(size); if (r == NULL) { printlogf(LOG_ERROR, "Out of memory!"); exit(LSYNCD_OUTOFMEMORY); } return r; } /** * "secured" calloc. */ void *s_calloc(size_t nmemb, size_t size) { void *r = calloc(nmemb, size); if (r == NULL) { printlogf(LOG_ERROR, "Out of memory!"); exit(LSYNCD_OUTOFMEMORY); } return r; } /** * "secured" realloc. */ void *s_realloc(void *ptr, size_t size) { void *r = realloc(ptr, size); if (r == NULL) { printlogf(LOG_ERROR, "Out of memory!"); exit(LSYNCD_OUTOFMEMORY); } return r; } /** * "secured" strdup. */ char *s_strdup(const char* src) { char *s = strdup(src); if (s == NULL) { printlogf(LOG_ERROR, "Out of memory!"); exit(LSYNCD_OUTOFMEMORY); } return s; } /** * Returns the canonicalized path of a directory with a final '/', * Makes sure it is a directory. */ char *realdir(const char * dir) { char* cs = s_malloc(PATH_MAX+1); cs = realpath(dir, cs); if (cs == NULL) { return NULL; } if (strlen(cs) + 1 >= PATH_MAX) { // at systems maxpath already, we cannot add a '/' anyway. return NULL; } struct stat st; stat(cs, &st); if (!S_ISDIR(st.st_mode)) { free(cs); return NULL; } strcat(cs, "/"); return cs; } /*--------------------------------------------------------------------------* * dir_configuration handling. *--------------------------------------------------------------------------*/ /** * (re)allocates space for a new dir_config and sets all values to 0/Null. * * (Yes we know, its a bit unoptimal, since when 6 dir_confs are given * in the config file, lsyncd will reallocate dir_confs 6 times. Well we * can live with that.) * * @return the pointer to the newly allocated dir_conf */ struct dir_conf * new_dir_conf() { if (dir_conf_n > 0) { dir_conf_n++; dir_confs = s_realloc(dir_confs, dir_conf_n * sizeof(struct dir_conf)); memset(&dir_confs[dir_conf_n - 1], 0, sizeof(struct dir_conf)); // creates targets NULL terminator (no targets yet) dir_confs[dir_conf_n - 1].targets = s_calloc(1, sizeof(char *)); return &dir_confs[dir_conf_n - 1]; } dir_conf_n++; dir_confs = s_calloc(dir_conf_n, sizeof(struct dir_conf)); // creates targets NULL terminator (no targets yet) dir_confs[0].targets = s_calloc(1, sizeof(char *)); return dir_confs; } /** * Adds a target to a dir_conf. target string will duped. * * @param dir_conf dir_conf to add the target to. * @param target target to add. */ void dir_conf_add_target(struct dir_conf * dir_conf, char *target) { char **t; int target_n = 0; /* count current targets */ for (t = dir_conf->targets; *t; ++t) { target_n++; } dir_conf->targets = s_realloc(dir_conf->targets, (target_n + 1) * sizeof(char *)); dir_conf->targets[target_n] = s_strdup(target); dir_conf->targets[target_n + 1] = NULL; } /*--------------------------------------------------------------------------* * ToSync Stack handling. *--------------------------------------------------------------------------*/ /** * Adds a directory to sync. * * @param watch the index in dir_watches to the directory. */ bool append_tosync_watch(int watch) { int i; printlogf(LOG_DEBUG, "append_tosync_watch(%d)", watch); // look if its already in the tosync list. for(i = 0; i < tosync_pos; i++) { if (tosync[i] == watch) { return true; } } if (tosync_pos + 1 >= tosync_size) { tosync_size *= 2; tosync = s_realloc(tosync, tosync_size*sizeof(int)); } tosync[tosync_pos++] = watch; return true; } /** * Removes a tosync entry in the stack at the position p. */ bool remove_tosync_pos(int p) { int i; assert(p < tosync_pos); //TODO improve performance by using memcpy. for(i = p; i < tosync_pos; i++) { tosync[i] = tosync[i + 1]; } tosync_pos--; return true; } /** * Parses an option text, replacing all '%' specifiers with * elaborated stuff. duh, currently there is only one, so this * fuction is a bit overkill but oh well :-) * * @param text string to parse. * @param recursive info for replacements. * * @return a newly allocated string. */ char *parse_option_text(char *text, bool recursive) { char * str = s_strdup(text); char * chr; // search result for %. // replace all '%' specifiers with there special meanings for(chr = strchr(str, '%'); chr; chr = strchr(str, '%')) { char *p; // chr points now to the '%' thing to be replaced switch (chr[1]) { case 'r' : // replace %r with 'r' when recursive or 'd' when not. chr[0] = recursive ? 'r' : 'd'; for(p = chr + 1; *p != 0; p++) { p[0] = p[1]; } break; case 0: // wtf, '%' was at the end of the string! default : // unknown char printlogf(LOG_ERROR, "don't know how to handle '\%' specifier in \"%s\"!", *text); exit(LSYNCD_BADPARAMETERS); } } return str; } /** * Calls the specified action (most likely rsync) to sync from src to dest. * Returns after the forked process has finished. * * @param dir_conf The config the is applicatable for this dir. * @param src Source string. * @param dest Destination string, * @param recursive If true -r will be handled on, -d (single directory) otherwise * @return true if successful, false if not. */ bool action(struct dir_conf * dir_conf, char const * src, const char * dest, bool recursive) { pid_t pid; int status; const int MAX_ARGS = 100; char * argv[MAX_ARGS]; int argc=0; int i; struct call_option* optp; optp = dir_conf->callopts ? dir_conf->callopts : default_callopts; argv[argc++] = s_strdup(dir_conf->binary ? dir_conf->binary : default_binary); for(;optp->kind != CO_EOL; optp++) { switch (optp->kind) { case CO_TEXT : argv[argc++] = parse_option_text(optp->text, recursive); continue; case CO_EXCLUDE : // --exclude-from and the exclude file // insert only when the exclude file is present otherwise skip it. if (dir_conf->exclude_file == NULL) { continue; } argv[argc++] = s_strdup("--exclude-from"); argv[argc++] = s_strdup(dir_conf->exclude_file); continue; case CO_SOURCE : argv[argc++] = s_strdup(src); continue; case CO_DEST : argv[argc++] = s_strdup(dest); continue; default: assert(false); } if (argc >= MAX_ARGS) { /* check for error condition */ printlogf(LOG_ERROR, "Internal error: too many (>%i) options passed", argc); return false; } } argv[argc++] = NULL; /* debug dump of command-line options */ //for (i=0; ibinary ? dir_conf->binary : default_binary; if (!flag_nodaemon) { freopen(logfile, "a", stdout); freopen(logfile, "a", stderr); } execv(binary, argv); // in a sane world does not return! printlogf(LOG_ERROR, "Failed executing [%s]", binary); exit(LSYNCD_INTERNALFAIL); } for (i=0; i [%s] finished", src, dest); return true; } /** * Adds a directory to watch * * @param pathname the absolute path of the directory to watch. * @param dirname the name of the directory only (yes this is a bit redudant, but oh well) * @param destname if not NULL call this dir that way on destionation. * @param parent if not -1 the index to the parent directory that is already watched * @param dir_conf the applicateable configuration * * @return index to dir_watches of the new dir, -1 on error. */ int add_watch(char const * pathname, char const * dirname, char const * destname, int parent, struct dir_conf * dir_conf) { int wd; int newdw; wd = inotify_add_watch(inotf, pathname, IN_ATTRIB | IN_CLOSE_WRITE | IN_CREATE | IN_DELETE | IN_DELETE_SELF | IN_MOVED_FROM | IN_MOVED_TO | IN_DONT_FOLLOW | IN_ONLYDIR); if (wd == -1) { printlogf(LOG_ERROR, "Cannot add watch %s (%d:%s)", pathname, errno, strerror(errno)); return -1; } // look if an unused slot can be found. for (newdw = 0; newdw < dir_watch_num; newdw++) { if (dir_watches[newdw].wd < 0) { break; } } if (newdw == dir_watch_num) { if (dir_watch_num + 1 >= dir_watch_size) { dir_watch_size *= 2; dir_watches = s_realloc(dir_watches, dir_watch_size * sizeof(struct dir_watch)); } dir_watch_num++; } dir_watches[newdw].wd = wd; dir_watches[newdw].parent = parent; dir_watches[newdw].dirname = s_strdup(dirname); dir_watches[newdw].destname = destname ? s_strdup(destname) : NULL; dir_watches[newdw].dir_conf = dir_conf; return newdw; } /** * Builds the abolute path name of a given directory beeing watched from the dir_watches information. * * @param pathname destination buffer to store the result to. * @param pathsize max size of this buffer * @param watch the index in dir_watches to the directory. * @param dirname if not NULL it is added at the end of pathname * @param prefix if not NULL it is added at the beginning of pathname */ bool buildpath(char *pathname, int pathsize, int watch, char const *dirname, char const *prefix) { int j, k, p, ps; pathname[0] = 0; if (prefix) { // make sure nobody calls me with %wwwuser option set assert(strcmp(prefix, "%wwwuser")); strcat(pathname, prefix); } // count how big the parent stack is for (p = watch, ps = 0; p != -1; p = dir_watches[p].parent, ps++) { } // now add the parent paths from back to front for (j = ps; j > 0; j--) { char * tmpname; // go j steps behind stack for (p = watch, k = 0; k + 1 < j; p = dir_watches[p].parent, k++) { } tmpname = (prefix && dir_watches[p].destname) ? dir_watches[p].destname : dir_watches[p].dirname; if (strlen(pathname) + strlen(tmpname) + 2 >= pathsize) { printlogf(LOG_ERROR, "path too long %s/...", tmpname); return false; } strcat(pathname, tmpname); strcat(pathname, "/"); } if (dirname) { if (strlen(pathname) + strlen(dirname) + 2 >= pathsize) { printlogf(LOG_ERROR, "path too long %s//%s", pathname, dirname); return false; } strcat(pathname, dirname); } return true; } /** * Syncs a directory. * * @param watch the index in dir_watches to the directory. * * @returns true when all targets were successful. */ bool rsync_dir(int watch) { char pathname[PATH_MAX+1]; char destname[PATH_MAX+1]; bool status = true; char ** target; if (!buildpath(pathname, sizeof(pathname), watch, NULL, NULL)) { return false; } for (target = dir_watches[watch].dir_conf->targets; *target; target++) { if (!buildpath(destname, sizeof(destname), watch, NULL, *target)) { status = false; continue; } printlogf(LOG_NORMAL, "rsyncing %s --> %s", pathname, destname); // call rsync to propagate changes in the directory if (!action(dir_watches[watch].dir_conf, pathname, destname, false)) { printlogf(LOG_ERROR, "Rsync from %s to %s failed", pathname, destname); status = false; } } return status; } /** * Adds a dir to watch. * * @param dirname The name or absolute path of the directory to watch. * @param destname If not NULL call this dir that way on sync destination. * @param parent If not -1, the index in dir_watches to the parent directory already watched. * Must have absolute path if parent == -1. * * @return the index in dir_watches off the directory or -1 on fail. * */ int add_dirwatch(char const * dirname, char const * destname, int parent, struct dir_conf * dir_conf) { DIR *d; struct dirent *de; int dw, i; char pathname[PATH_MAX+1]; printlogf(LOG_DEBUG, "add_dirwatch(%s, %s, p->dirname:%s, ...)", dirname, destname, parent >= 0 ? dir_watches[parent].dirname : "NULL"); if (!buildpath(pathname, sizeof(pathname), parent, dirname, NULL)) { return -1; } for (i = 0; i < exclude_dir_n; i++) { if (!strcmp(dirname, exclude_dirs[i])) { return -1; } } dw = add_watch(pathname, dirname, destname, parent, dir_conf); if (dw == -1) { return -1; } if (strlen(pathname) + strlen(dirname) + 2 > sizeof(pathname)) { printlogf(LOG_ERROR, "pathname too long %s//%s", pathname, dirname); return -1; } d = opendir(pathname); if (d == NULL) { printlogf(LOG_ERROR, "cannot open dir %s.", dirname); return -1; } while (keep_going) { struct stat st; char subdir[PATH_MAX+1]; bool isdir; de = readdir(d); if (de == NULL) { break; } if (de->d_type == DT_DIR) { isdir = true; } else if (de->d_type == DT_UNKNOWN) { // in case of reiserfs, d_type will be UNKNOWN, how evil! :-( // use traditional means to determine if its a directory. isdir = buildpath(subdir, sizeof(subdir), dw, de->d_name, NULL) && !stat(subdir, &st) && S_ISDIR(st.st_mode); } else { isdir = false; } if (isdir && strcmp(de->d_name, "..") && strcmp(de->d_name, ".")) { int ndw = add_dirwatch(de->d_name, NULL, dw, dir_conf); printlogf(LOG_NORMAL, "found new directory: %s/%s -- added on tosync stack.", dirname, de->d_name); append_tosync_watch(ndw); } } closedir(d); return dw; } /** * Removes a watched dir, including recursevily all subdirs. * * @param name Optionally. If not NULL, the directory name to remove which is a child of parent. * @param parent The index to the parent directory of the directory 'name' to remove, * or to be removed itself if name == NULL. */ bool remove_dirwatch(const char * name, int parent) { int i; int dw; if (name) { // look for the child with the name for (i = 0; i < dir_watch_num; i++) { if (dir_watches[i].wd >= 0 && dir_watches[i].parent == parent && !strcmp(name, dir_watches[i].dirname)) { dw = i; break; } } if (i >= dir_watch_num) { printlogf(LOG_ERROR, "Cannot find entry for %s:/:%s :-(", dir_watches[parent].dirname, name); return false; } } else { dw = parent; } for (i = 0; i < dir_watch_num; i++) { if (dir_watches[i].wd >= 0 && dir_watches[i].parent == dw) { remove_dirwatch(NULL, i); } } inotify_rm_watch(inotf, dir_watches[dw].wd); dir_watches[dw].wd = -1; free(dir_watches[dw].dirname); dir_watches[dw].dirname = NULL; if (dir_watches[dw].destname) { free(dir_watches[dw].destname); dir_watches[dw].destname = NULL; } return true; } /** * Find the matching dw entry from wd (watch descriptor), and return * the offset in the table. * * @param wd The wd (watch descriptor) given by inotify * @return offset, or -1 if not found */ int get_dirwatch_offset(int wd) { int i; for (i = 0; i < dir_watch_num; i++) { if (dir_watches[i].wd == wd) { break; } } if (i >= dir_watch_num) { return -1; } else { return i; } } /** * Processes through the tosync stack, rysncing all its directories. * * TODO: make special logic to determine who is a subdirectory of whom, and maybe optimizie calls. */ bool process_tosync_stack() { printlogf(LOG_DEBUG, "Processing through tosync stack."); while(tosync_pos > 0) { rsync_dir(tosync[--tosync_pos]); } printlogf(LOG_DEBUG, "being done with tosync stack"); return true; } /** * Handles an inotify event. * * @param event The event to handle */ bool handle_event(struct inotify_event *event) { char masktext[255] = {0,}; int mask = event->mask; int i, watch; int subwatch = -1; struct inotify_mask_text *p; for (p = mask_texts; p->mask; p++) { if (mask & p->mask) { if (strlen(masktext) + strlen(p->text) + 3 >= sizeof(masktext)) { printlogf(LOG_ERROR, "bufferoverflow in handle_event"); return false; } if (*masktext) { strcat(masktext, ", "); } strcat(masktext, p->text); } } printlogf(LOG_DEBUG, "inotfy event: %s:%s", masktext, event->name); if (IN_IGNORED & event->mask) { return true; } for (i = 0; i < exclude_dir_n; i++) { if (!strcmp(event->name, exclude_dirs[i])) { return true; } } watch = get_dirwatch_offset(event->wd); if (watch == -1) { printlogf(LOG_ERROR, "received an inotify event that doesnt match any watched directory :-(%d,%d)", event->mask, event->wd); return false; } if (((IN_CREATE | IN_MOVED_TO) & event->mask) && (IN_ISDIR & event->mask)) { subwatch = add_dirwatch(event->name, NULL, watch, dir_watches[watch].dir_conf); } if (((IN_DELETE | IN_MOVED_FROM) & event->mask) && (IN_ISDIR & event->mask)) { remove_dirwatch(event->name, watch); } if ((IN_CREATE | IN_CLOSE_WRITE | IN_DELETE | IN_MOVED_TO | IN_MOVED_FROM) & event->mask ) { printlogf(LOG_NORMAL, "event %s:%s triggered.", masktext, event->name); rsync_dir(watch); // TODO, worry about errors? if (subwatch >= 0) { // sync through the new created directory as well. rsync_dir(subwatch); } } else { printlogf(LOG_DEBUG, "... ignored this event."); } process_tosync_stack(); return true; } /** * The control loop waiting for inotify events. */ bool master_loop() { char buf[INOTIFY_BUF_LEN]; int len, i = 0; while (keep_going) { len = read (inotf, buf, INOTIFY_BUF_LEN); if (len < 0) { printlogf(LOG_ERROR, "failed to read from inotify (%d:%s)", errno, strerror(errno)); return false; } if (len == 0) { printlogf(LOG_ERROR, "eof?"); return false; } i = 0; while (i < len) { struct inotify_event *event = (struct inotify_event *) &buf[i]; handle_event(event); i += sizeof(struct inotify_event) + event->len; } } return true; } /** * Utility function to check file exists. Print out error message and die. * * @param filename filename to check */ void check_file_exists(const char* filename) { struct stat st; if (-1==stat(filename, &st)) { printlogf(LOG_ERROR, "File [%s] does not exist\n", filename); exit (LSYNCD_FILENOTFOUND); } } /** * Utility function to check given path is absolute path. * * @param filename filename to check */ void check_absolute_path(const char* filename) { if (filename[0] != '/') { printlogf(LOG_ERROR, "Filename [%s] is not an absolute path\n", filename); exit (LSYNCD_FILENOTFOUND); } } /** * Prints the help text and exits 0. * * @param arg0 argv[0] to show what lsyncd was called with. */ void print_help(char *arg0) { printf("\n"); #ifdef XML_CONFIG printf("USAGE: %s [OPTION]... [SOURCE] [TARGET 1] [TARGET 2] ...\n", arg0); #else printf("USAGE: %s [OPTION]... SOURCE TARGET-1 TARGET-2 ...\n", arg0); #endif printf("\n"); printf("SOURCE: a directory to watch and rsync.\n"); printf("\n"); printf("TARGET: can be any name accepted by rsync. e.g. \"foohost::barmodule/\"\n"); printf("\n"); #ifdef XML_CONFIG printf("When called without SOURCE and TARGET, the\n"); printf("configuration will be read from the config file\"\n"); #endif printf("\n"); printf("OPTIONS:\n"); printf(" --binary FILE Call this binary to sync (DEFAULT: %s)\n", default_binary); #ifdef XML_CONFIG printf(" --conf FILE Load configuration from this file\n"); printf(" (DEFAULT: %s if called without SOURCE/TARGET)\n", conf_filename); #endif printf(" --debug Log debug messages\n"); printf(" --dryrun Do not call any actions, run dry only\n"); printf(" --exclude-from FILE Exclude file handled to rsync (DEFAULT: None)\n"); printf(" --help Print this help text and exit.\n"); printf(" --logfile FILE Put log here (DEFAULT: %s)\n", logfile); printf(" --no-daemon Do not detach, log to stdout/stderr\n"); printf(" --pidfile FILE Create a file containing pid of the daemon\n"); printf(" --scarce Only log errors\n"); printf(" --version Print version an exit.\n"); printf("\n"); printf("Take care that lsyncd is allowed to write to the logfile specified.\n"); printf("\n"); printf("EXCLUDE FILE: \n"); printf(" The exclude file may have either filebased general masks like \"*.php\" without directory specifications,\n"); printf(" or exclude complete directories like \"Data/\". lsyncd will recognize directory excludes by the trailing '/'\n"); printf(" and will not add watches of directories of exactly such name including sub-directories of them.\n"); printf(" Please do not try to use more sophisticated exclude masks like \"Data/*.dat\" or \"Da*a/\", \"Data/Volatile/\" etc.\n"); printf(" This will not work like you would expect it to.\n"); printf("\n"); printf("LICENSE\n"); printf(" GPLv2 or any later version. See COPYING\n"); printf("\n"); #ifndef XML_CONFIG printf("(this lsyncd binary was not compiled to be able to read config files)\n"); #endif exit(0); } #ifdef XML_CONFIG /*--------------------------------------------------------------------------* * config file parsing *--------------------------------------------------------------------------*/ /** * Parses * * @return the allocated and filled calloptions structure */ struct call_option * parse_callopts(xmlNodePtr node) { xmlNodePtr cnode; xmlChar *xc; int opt_n = 0; struct call_option * asw; // count how many options are there for (cnode = node->children; cnode; cnode = cnode->next) { if (cnode->type != XML_ELEMENT_NODE) { continue; } if (xmlStrcmp(cnode->name, BAD_CAST "option") && xmlStrcmp(cnode->name, BAD_CAST "exclude-file") && xmlStrcmp(cnode->name, BAD_CAST "source") && xmlStrcmp(cnode->name, BAD_CAST "destination") ) { fprintf(stderr, "error unknown call option type \"%s\"", cnode->name); exit(LSYNCD_BADCONFIGFILE); } opt_n++; } opt_n++; asw = (struct call_option *) s_calloc(opt_n, sizeof(struct call_option)); // fill in the answer opt_n = 0; for (cnode = node->children; cnode; cnode = cnode->next) { if (cnode->type != XML_ELEMENT_NODE) { continue; } asw[opt_n].text = NULL; if (!xmlStrcmp(cnode->name, BAD_CAST "option")) { xc = xmlGetProp(cnode, BAD_CAST "text"); if (xc == NULL) { fprintf(stderr, "error in config file: text attribute missing from