/* * libcsync -- a library to sync a directory with another * * Copyright (c) 2008 by Andreas Schneider * * 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., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. */ #include "config.h" #ifndef _GNU_SOURCE #define _GNU_SOURCE #endif #include #include #include #include #include #include #include #include "c_lib.h" #include "csync_private.h" #include "csync_config.h" #include "csync_exclude.h" #include "csync_lock.h" #include "csync_statedb.h" #include "csync_time.h" #include "csync_util.h" #include "csync_misc.h" #include "csync_update.h" #include "csync_reconcile.h" #include "csync_propagate.h" #include "vio/csync_vio.h" #define CSYNC_LOG_CATEGORY_NAME "csync.api" #include "csync_log.h" static enum csync_error_codes_e _csync_errno; static int _key_cmp(const void *key, const void *data) { uint64_t a; csync_file_stat_t *b; a = *(uint64_t *) (key); b = (csync_file_stat_t *) data; if (a < b->phash) { return -1; } else if (a > b->phash) { return 1; } return 0; } static int _data_cmp(const void *key, const void *data) { csync_file_stat_t *a, *b; a = (csync_file_stat_t *) key; b = (csync_file_stat_t *) data; if (a->phash < b->phash) { return -1; } else if (a->phash > b->phash) { return 1; } return 0; } int csync_create(CSYNC **csync, const char *local, const char *remote) { CSYNC *ctx; size_t len = 0; char *home; int rc; _csync_errno = CSYNC_ERR_NONE; ctx = c_malloc(sizeof(CSYNC)); if (ctx == NULL) { _csync_errno = CSYNC_ERR_MEM; return -1; } /* remove trailing slashes */ len = strlen(local); while(len > 0 && local[len - 1] == '/') --len; ctx->local.uri = c_strndup(local, len); if (ctx->local.uri == NULL) { _csync_errno = CSYNC_ERR_MEM; return -1; } /* remove trailing slashes */ len = strlen(remote); while(len > 0 && remote[len - 1] == '/') --len; ctx->remote.uri = c_strndup(remote, len); if (ctx->remote.uri == NULL) { SAFE_FREE(ctx->remote.uri); _csync_errno = CSYNC_ERR_MEM; return -1; } ctx->options.max_depth = MAX_DEPTH; ctx->options.max_time_difference = MAX_TIME_DIFFERENCE; ctx->options.unix_extensions = 0; ctx->options.with_conflict_copys=false; ctx->options.local_only_mode = false; ctx->pwd.uid = getuid(); ctx->pwd.euid = geteuid(); home = csync_get_user_home_dir(); if (home == NULL) { SAFE_FREE(ctx->local.uri); SAFE_FREE(ctx->remote.uri); SAFE_FREE(ctx); errno = ENOMEM; _csync_errno = CSYNC_ERR_MEM; return -1; } rc = asprintf(&ctx->options.config_dir, "%s/%s", home, CSYNC_CONF_DIR); SAFE_FREE(home); if (rc < 0) { SAFE_FREE(ctx->local.uri); SAFE_FREE(ctx->remote.uri); SAFE_FREE(ctx); errno = ENOMEM; _csync_errno = CSYNC_ERR_MEM; return -1; } *csync = ctx; return 0; } int csync_init(CSYNC *ctx) { int rc; time_t timediff = -1; char *log = NULL; char *exclude = NULL; char *lock = NULL; char *config = NULL; #ifndef _WIN32 char errbuf[256] = {0}; #endif _csync_errno = CSYNC_ERR_NONE; if (ctx == NULL) { _csync_errno = CSYNC_ERR_UNSPEC; errno = EBADF; return -1; } /* Do not initialize twice */ if (ctx->status & CSYNC_STATUS_INIT) { return 1; } /* load log file */ if (csync_log_init() < 0) { _csync_errno = CSYNC_ERR_LOG; fprintf(stderr, "csync_init: logger init failed\n"); return -1; } /* create dir if it doesn't exist */ if (! c_isdir(ctx->options.config_dir)) { c_mkdirs(ctx->options.config_dir, 0700); } if (asprintf(&log, "%s/%s", ctx->options.config_dir, CSYNC_LOG_FILE) < 0) { _csync_errno = CSYNC_ERR_UNSPEC; rc = -1; goto out; } /* load log if it exists */ if (c_isfile(log)) { csync_log_load(log); } else { #ifndef _WIN32 if (c_copy(SYSCONFDIR "/csync/" CSYNC_LOG_FILE, log, 0644) == 0) { csync_log_load(log); } #endif } /* create lock file */ if (asprintf(&lock, "%s/%s", ctx->options.config_dir, CSYNC_LOCK_FILE) < 0) { _csync_errno = CSYNC_ERR_UNSPEC; rc = -1; goto out; } #ifndef _WIN32 if (csync_lock(lock) < 0) { _csync_errno = CSYNC_ERR_LOCK; rc = -1; goto out; } #endif /* load config file */ if (asprintf(&config, "%s/%s", ctx->options.config_dir, CSYNC_CONF_FILE) < 0) { rc = -1; goto out; } if (csync_config_load(ctx, config) < 0) { CSYNC_LOG(CSYNC_LOG_PRIORITY_WARN, "Could not load config file %s, using defaults.", config); } #ifndef _WIN32 /* load global exclude list */ if (asprintf(&exclude, "%s/csync/%s", SYSCONFDIR, CSYNC_EXCLUDE_FILE) < 0) { _csync_errno = CSYNC_ERR_UNSPEC; rc = -1; goto out; } if (csync_exclude_load(ctx, exclude) < 0) { strerror_r(errno, errbuf, sizeof(errbuf)); CSYNC_LOG(CSYNC_LOG_PRIORITY_INFO, "Could not load %s - %s", exclude, errbuf); } SAFE_FREE(exclude); /* load exclude list */ if (asprintf(&exclude, "%s/%s", ctx->options.config_dir, CSYNC_EXCLUDE_FILE) < 0) { _csync_errno = CSYNC_ERR_UNSPEC; rc = -1; goto out; } if (csync_exclude_load(ctx, exclude) < 0) { strerror_r(errno, errbuf, sizeof(errbuf)); CSYNC_LOG(CSYNC_LOG_PRIORITY_INFO, "Could not load %s - %s", exclude, errbuf); } #endif /* create/load statedb */ if (! csync_is_statedb_disabled(ctx)) { uint64_t h = csync_create_statedb_hash(ctx); if (asprintf(&ctx->statedb.file, "%s/csync_statedb_%llu.db", ctx->options.config_dir, (long long unsigned int) h) < 0) { _csync_errno = CSYNC_ERR_UNSPEC; rc = -1; goto out; } CSYNC_LOG(CSYNC_LOG_PRIORITY_DEBUG, "Remote replica: %s", ctx->remote.uri); CSYNC_LOG(CSYNC_LOG_PRIORITY_DEBUG, "Statedb: %s", ctx->statedb.file); if (csync_statedb_load(ctx, ctx->statedb.file) < 0) { _csync_errno = CSYNC_ERR_STATEDB_LOAD; rc = -1; goto out; } } ctx->local.type = LOCAL_REPLICA; /* check for uri */ if ( !ctx->options.local_only_mode && csync_fnmatch("*://*", ctx->remote.uri, 0) == 0) { size_t len; len = strstr(ctx->remote.uri, "://") - ctx->remote.uri; /* get protocol */ if (len > 0) { char *module = NULL; /* module name */ module = c_strndup(ctx->remote.uri, len); if (module == NULL) { _csync_errno = CSYNC_ERR_MODULE; rc = -1; goto out; } /* load module */ retry_vio_init: rc = csync_vio_init(ctx, module, NULL); if (rc < 0) { len = strlen(module); if (module[len-1] == 's') { module[len-1] = '\0'; goto retry_vio_init; } SAFE_FREE(module); goto out; } SAFE_FREE(module); ctx->remote.type = REMOTE_REPLCIA; } } else { ctx->remote.type = LOCAL_REPLICA; } if( !ctx->options.local_only_mode ) { timediff = csync_timediff(ctx); if (timediff > ctx->options.max_time_difference) { CSYNC_LOG(CSYNC_LOG_PRIORITY_FATAL, "Clock skew detected. The time difference is greater than %d seconds!", ctx->options.max_time_difference); _csync_errno = CSYNC_ERR_TIMESKEW; rc = -1; goto out; } else if (timediff < 0) { CSYNC_LOG(CSYNC_LOG_PRIORITY_FATAL, "Synchronisation is not possible!"); _csync_errno = csync_time_errno(); rc = -1; goto out; } if (csync_unix_extensions(ctx) < 0) { CSYNC_LOG(CSYNC_LOG_PRIORITY_FATAL, "Could not detect filesystem type."); _csync_errno = CSYNC_ERR_FILESYSTEM; rc = -1; goto out; } } if (c_rbtree_create(&ctx->local.tree, _key_cmp, _data_cmp) < 0) { _csync_errno = CSYNC_ERR_TREE; rc = -1; goto out; } if (c_rbtree_create(&ctx->remote.tree, _key_cmp, _data_cmp) < 0) { _csync_errno = CSYNC_ERR_TREE; rc = -1; goto out; } ctx->status = CSYNC_STATUS_INIT; rc = 0; out: SAFE_FREE(log); SAFE_FREE(lock); SAFE_FREE(exclude); SAFE_FREE(config); return rc; } int csync_update(CSYNC *ctx) { int rc = -1; struct timespec start, finish; _csync_errno = CSYNC_ERR_NONE; if (ctx == NULL) { errno = EBADF; _csync_errno = CSYNC_ERR_PARAM; return -1; } csync_memstat_check(); /* update detection for local replica */ csync_gettime(&start); ctx->current = LOCAL_REPLICA; ctx->replica = ctx->local.type; rc = csync_ftw(ctx, ctx->local.uri, csync_walker, MAX_DEPTH); csync_gettime(&finish); CSYNC_LOG(CSYNC_LOG_PRIORITY_DEBUG, "Update detection for local replica took %.2f seconds walking %zu files.", c_secdiff(finish, start), c_rbtree_size(ctx->local.tree)); csync_memstat_check(); if (rc < 0) { _csync_errno = CSYNC_ERR_TREE; return -1; } /* update detection for remote replica */ if( ! ctx->options.local_only_mode ) { csync_gettime(&start); ctx->current = REMOTE_REPLCIA; ctx->replica = ctx->remote.type; rc = csync_ftw(ctx, ctx->remote.uri, csync_walker, MAX_DEPTH); csync_gettime(&finish); CSYNC_LOG(CSYNC_LOG_PRIORITY_DEBUG, "Update detection for remote replica took %.2f seconds " "walking %zu files.", c_secdiff(finish, start), c_rbtree_size(ctx->remote.tree)); csync_memstat_check(); if (rc < 0) { _csync_errno = CSYNC_ERR_TREE; return -1; } } ctx->status |= CSYNC_STATUS_UPDATE; return 0; } int csync_reconcile(CSYNC *ctx) { int rc = -1; struct timespec start, finish; _csync_errno = CSYNC_ERR_NONE; if (ctx == NULL) { _csync_errno = CSYNC_ERR_PARAM; errno = EBADF; return -1; } /* Reconciliation for local replica */ csync_gettime(&start); ctx->current = LOCAL_REPLICA; ctx->replica = ctx->local.type; rc = csync_reconcile_updates(ctx); csync_gettime(&finish); CSYNC_LOG(CSYNC_LOG_PRIORITY_DEBUG, "Reconciliation for local replica took %.2f seconds visiting %zu files.", c_secdiff(finish, start), c_rbtree_size(ctx->local.tree)); if (rc < 0) { _csync_errno = CSYNC_ERR_RECONCILE; return -1; } /* Reconciliation for local replica */ csync_gettime(&start); ctx->current = REMOTE_REPLCIA; ctx->replica = ctx->remote.type; rc = csync_reconcile_updates(ctx); csync_gettime(&finish); CSYNC_LOG(CSYNC_LOG_PRIORITY_DEBUG, "Reconciliation for remote replica took %.2f seconds visiting %zu files.", c_secdiff(finish, start), c_rbtree_size(ctx->remote.tree)); if (rc < 0) { _csync_errno = CSYNC_ERR_RECONCILE; return -1; } ctx->status |= CSYNC_STATUS_RECONCILE; return 0; } int csync_propagate(CSYNC *ctx) { int rc = -1; struct timespec start, finish; _csync_errno = CSYNC_ERR_NONE; if (ctx == NULL) { _csync_errno = CSYNC_ERR_PARAM; errno = EBADF; return -1; } /* Reconciliation for local replica */ csync_gettime(&start); ctx->current = LOCAL_REPLICA; ctx->replica = ctx->local.type; rc = csync_propagate_files(ctx); csync_gettime(&finish); CSYNC_LOG(CSYNC_LOG_PRIORITY_DEBUG, "Propagation for local replica took %.2f seconds visiting %zu files.", c_secdiff(finish, start), c_rbtree_size(ctx->local.tree)); if (rc < 0) { _csync_errno = CSYNC_ERR_PROPAGATE; return -1; } /* Reconciliation for local replica */ csync_gettime(&start); ctx->current = REMOTE_REPLCIA; ctx->replica = ctx->remote.type; rc = csync_propagate_files(ctx); csync_gettime(&finish); CSYNC_LOG(CSYNC_LOG_PRIORITY_DEBUG, "Propagation for remote replica took %.2f seconds visiting %zu files.", c_secdiff(finish, start), c_rbtree_size(ctx->remote.tree)); if (rc < 0) { _csync_errno = CSYNC_ERR_PROPAGATE; return -1; } ctx->status |= CSYNC_STATUS_PROPAGATE; return 0; } /* * local visitor which calls the user visitor with repacked stat info. */ static int _csync_treewalk_visitor( void *obj, void *data ) { csync_file_stat_t *cur = NULL; CSYNC *ctx = NULL; c_rbtree_visit_func *visitor = NULL; _csync_treewalk_context *twctx = NULL; TREE_WALK_FILE trav; _csync_errno = CSYNC_ERR_NONE; cur = (csync_file_stat_t *) obj; ctx = (CSYNC *) data; if( ctx ) twctx = (_csync_treewalk_context*) ctx->userdata; if( twctx->instruction_filter > 0 && !(twctx->instruction_filter & cur->instruction) ) { return 0; } if(cur && twctx) { visitor = (c_rbtree_visit_func*)(twctx->user_visitor); if(visitor) { trav.path = cur->path; trav.modtime = cur->modtime; trav.uid = cur->uid; trav.gid = cur->gid; trav.mode = cur->mode; trav.type = cur->type; trav.instruction = cur->instruction; return (*visitor)(&trav, twctx->userdata); } } _csync_errno = CSYNC_ERR_TREE; return -1; } /* * treewalk function, called from its wrappers below. * * it encapsulates the user visitor function, the filter and the userdata * into a treewalk_context structure and calls the rb treewalk function, * which calls the local _csync_treewalk_visitor in this module. * The user visitor is called from there. */ static int _csync_walk_tree(CSYNC *ctx, c_rbtree_t *tree, csync_treewalk_visit_func *visitor, int filter) { _csync_treewalk_context tw_ctx; int rc = -1; _csync_errno = CSYNC_ERR_NONE; if( !(visitor && tree && ctx)) { _csync_errno = CSYNC_ERR_PARAM; return rc; } tw_ctx.userdata = ctx->userdata; tw_ctx.user_visitor = visitor; tw_ctx.instruction_filter = filter; ctx->userdata = &tw_ctx; rc = c_rbtree_walk(tree, (void*) ctx, _csync_treewalk_visitor); if( rc < 0 ) _csync_errno = CSYNC_ERR_TREE; ctx->userdata = tw_ctx.userdata; return rc; } /* * wrapper function for treewalk on the remote tree */ int csync_walk_remote_tree(CSYNC *ctx, csync_treewalk_visit_func *visitor, int filter) { c_rbtree_t *tree = NULL; int rc = -1; _csync_errno = CSYNC_ERR_NONE; if(ctx) { tree = ctx->remote.tree; } rc = _csync_walk_tree(ctx, tree, visitor, filter); if(rc < 0) _csync_errno = CSYNC_ERR_TREE; return rc; } /* * wrapper function for treewalk on the local tree */ int csync_walk_local_tree(CSYNC *ctx, csync_treewalk_visit_func *visitor, int filter) { c_rbtree_t *tree = NULL; int rc = -1; _csync_errno = CSYNC_ERR_NONE; if(ctx) { tree = ctx->local.tree; } rc = _csync_walk_tree(ctx, tree, visitor, filter); if(rc < 0) _csync_errno = CSYNC_ERR_TREE; return rc; } static void _tree_destructor(void *data) { csync_file_stat_t *freedata = NULL; _csync_errno = CSYNC_ERR_NONE; freedata = (csync_file_stat_t *) data; SAFE_FREE(freedata); } int csync_destroy(CSYNC *ctx) { struct timespec start, finish; char *lock = NULL; char errbuf[256] = {0}; int jwritten = 0; _csync_errno = CSYNC_ERR_NONE; if (ctx == NULL) { _csync_errno = CSYNC_ERR_PARAM; errno = EBADF; return -1; } csync_vio_shutdown(ctx); /* if we have a statedb */ if (ctx->statedb.db != NULL) { /* and we have successfully synchronized */ if (ctx->status >= CSYNC_STATUS_DONE) { /* merge trees */ if (csync_merge_file_trees(ctx) < 0) { strerror_r(errno, errbuf, sizeof(errbuf)); CSYNC_LOG(CSYNC_LOG_PRIORITY_ERROR, "Unable to merge trees: %s", errbuf); } else { csync_gettime(&start); /* write the statedb to disk */ if (csync_statedb_write(ctx) == 0) { jwritten = 1; csync_gettime(&finish); CSYNC_LOG(CSYNC_LOG_PRIORITY_DEBUG, "Writing the statedb of %zu files to disk took %.2f seconds", c_rbtree_size(ctx->local.tree), c_secdiff(finish, start)); } else { strerror_r(errno, errbuf, sizeof(errbuf)); CSYNC_LOG(CSYNC_LOG_PRIORITY_ERROR, "Unable to write statedb: %s", errbuf); } } } csync_statedb_close(ctx, ctx->statedb.file, jwritten); } /* clear exclude list */ csync_exclude_destroy(ctx); #ifndef _WIN32 /* remove the lock file */ if (asprintf(&lock, "%s/%s", ctx->options.config_dir, CSYNC_LOCK_FILE) > 0) { csync_lock_remove(lock); } #endif /* stop logging */ csync_log_fini(); /* destroy the rbtrees */ if (c_rbtree_size(ctx->local.tree) > 0) { c_rbtree_destroy(ctx->local.tree, _tree_destructor); } if (c_rbtree_size(ctx->remote.tree) > 0) { c_rbtree_destroy(ctx->remote.tree, _tree_destructor); } /* free memory */ c_rbtree_free(ctx->local.tree); c_list_free(ctx->local.list); c_rbtree_free(ctx->remote.tree); c_list_free(ctx->remote.list); SAFE_FREE(ctx->local.uri); SAFE_FREE(ctx->remote.uri); SAFE_FREE(ctx->options.config_dir); SAFE_FREE(ctx->statedb.file); SAFE_FREE(ctx); SAFE_FREE(lock); return 0; } /* Check if csync is the required version or get the version string. */ const char *csync_version(int req_version) { _csync_errno = CSYNC_ERR_NONE; if (req_version <= LIBCSYNC_VERSION_INT) { return CSYNC_STRINGIFY(LIBCSYNC_VERSION); } return NULL; } int csync_add_exclude_list(CSYNC *ctx, const char *path) { _csync_errno = CSYNC_ERR_NONE; if (ctx == NULL || path == NULL) { return -1; } return csync_exclude_load(ctx, path); } const char *csync_get_config_dir(CSYNC *ctx) { _csync_errno = CSYNC_ERR_NONE; if (ctx == NULL) { return NULL; } return ctx->options.config_dir; } int csync_set_config_dir(CSYNC *ctx, const char *path) { _csync_errno = CSYNC_ERR_NONE; if (ctx == NULL || path == NULL) { _csync_errno = CSYNC_ERR_PARAM; return -1; } SAFE_FREE(ctx->options.config_dir); ctx->options.config_dir = c_strdup(path); if (ctx->options.config_dir == NULL) { _csync_errno = CSYNC_ERR_MEM; return -1; } return 0; } int csync_enable_statedb(CSYNC *ctx) { _csync_errno = CSYNC_ERR_NONE; if (ctx == NULL) { _csync_errno = CSYNC_ERR_MEM; return -1; } if (ctx->status & CSYNC_STATUS_INIT) { _csync_errno = CSYNC_ERR_UNSPEC; fprintf(stderr, "This function must be called before initialization."); return -1; } ctx->statedb.disabled = 0; return 0; } int csync_disable_statedb(CSYNC *ctx) { _csync_errno = CSYNC_ERR_NONE; if (ctx == NULL) { _csync_errno = CSYNC_ERR_PARAM; return -1; } if (ctx->status & CSYNC_STATUS_INIT) { _csync_errno = CSYNC_ERR_UNSPEC; fprintf(stderr, "This function must be called before initialization."); return -1; } ctx->statedb.disabled = 1; return 0; } int csync_is_statedb_disabled(CSYNC *ctx) { _csync_errno = CSYNC_ERR_NONE; if (ctx == NULL) { _csync_errno = CSYNC_ERR_PARAM; return -1; } return ctx->statedb.disabled; } int csync_set_auth_callback(CSYNC *ctx, csync_auth_callback cb) { _csync_errno = CSYNC_ERR_NONE; if (ctx == NULL || cb == NULL) { _csync_errno = CSYNC_ERR_PARAM; return -1; } if (ctx->status & CSYNC_STATUS_INIT) { fprintf(stderr, "This function must be called before initialization."); _csync_errno = CSYNC_ERR_UNSPEC; return -1; } ctx->auth_callback = cb; return 0; } const char *csync_get_statedb_file(CSYNC *ctx) { _csync_errno = CSYNC_ERR_NONE; if (ctx == NULL) { _csync_errno = CSYNC_ERR_PARAM; return NULL; } return c_strdup(ctx->statedb.file); } void *csync_get_userdata(CSYNC *ctx) { _csync_errno = CSYNC_ERR_NONE; if (ctx == NULL) { _csync_errno = CSYNC_ERR_PARAM; return NULL; } return ctx->userdata; } int csync_set_userdata(CSYNC *ctx, void *userdata) { _csync_errno = CSYNC_ERR_NONE; if (ctx == NULL) { _csync_errno = CSYNC_ERR_PARAM; return -1; } ctx->userdata = userdata; return 0; } csync_auth_callback csync_get_auth_callback(CSYNC *ctx) { _csync_errno = CSYNC_ERR_NONE; if (ctx == NULL) { _csync_errno = CSYNC_ERR_PARAM; return NULL; } return ctx->auth_callback; } int csync_set_status(CSYNC *ctx, int status) { _csync_errno = CSYNC_ERR_NONE; if (ctx == NULL || status < 0) { _csync_errno = CSYNC_ERR_PARAM; return -1; } ctx->status = status; return 0; } int csync_get_status(CSYNC *ctx) { _csync_errno = CSYNC_ERR_NONE; if (ctx == NULL) { _csync_errno = CSYNC_ERR_PARAM; return -1; } return ctx->status; } int csync_enable_conflictcopys(CSYNC* ctx){ _csync_errno = CSYNC_ERR_NONE; if (ctx == NULL) { _csync_errno = CSYNC_ERR_PARAM; return -1; } if (ctx->status & CSYNC_STATUS_INIT) { fprintf(stderr, "This function must be called before initialization."); _csync_errno = CSYNC_ERR_UNSPEC; return -1; } ctx->options.with_conflict_copys=true; return 0; } int csync_set_local_only( CSYNC *ctx, bool local_only ) { _csync_errno = CSYNC_ERR_NONE; if (ctx == NULL) { _csync_errno = CSYNC_ERR_PARAM; return -1; } if (ctx->status & CSYNC_STATUS_INIT) { fprintf(stderr, "This function must be called before initialization."); _csync_errno = CSYNC_ERR_UNSPEC; return -1; } ctx->options.local_only_mode=local_only; return 0; } bool csync_get_local_only( CSYNC *ctx ) { _csync_errno = CSYNC_ERR_NONE; if (ctx == NULL) { _csync_errno = CSYNC_ERR_PARAM; return -1; } return ctx->options.local_only_mode; } CSYNC_ERROR_CODE csync_errno(void) { return _csync_errno; } /* vim: set ts=8 sw=2 et cindent: */