nextcloud-desktop/src/csync.c

1246 lines
29 KiB
C
Raw Normal View History

2008-02-27 20:56:47 +03:00
/*
* libcsync -- a library to sync a directory with another
*
2012-10-27 17:15:59 +04:00
* Copyright (c) 2008-2012 by Andreas Schneider <asn@cryptomilk.org>
2008-02-27 20:56:47 +03:00
*
* 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.
2008-02-27 20:56:47 +03:00
*
* 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.
2008-02-27 20:56:47 +03:00
*
* 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.
2008-02-27 20:56:47 +03:00
*/
2012-03-02 19:47:34 +04:00
#include "config.h"
#ifndef _GNU_SOURCE
#define _GNU_SOURCE
#endif
2008-04-28 14:40:32 +04:00
2008-02-27 20:56:47 +03:00
#include <errno.h>
#include <stdio.h>
#include <string.h>
#include <time.h>
#include <unistd.h>
#include <sys/types.h>
#include <stdbool.h>
2008-02-27 20:56:47 +03:00
#ifdef WITH_ICONV
#include <iconv.h>
#endif
2008-02-27 20:56:47 +03:00
#include "c_lib.h"
#include "csync_private.h"
2008-03-06 19:43:58 +03:00
#include "csync_config.h"
#include "csync_exclude.h"
#include "csync_lock.h"
#include "csync_statedb.h"
#include "csync_time.h"
#include "csync_util.h"
2012-03-02 16:38:39 +04:00
#include "csync_misc.h"
#include "c_jhash.h"
#include "std/c_private.h"
2008-02-27 20:56:47 +03:00
#include "csync_update.h"
2008-05-15 15:50:34 +04:00
#include "csync_reconcile.h"
#include "csync_propagate.h"
#include "vio/csync_vio.h"
2008-02-27 20:56:47 +03:00
#include "csync_log.h"
2013-01-04 23:45:10 +04:00
#include "csync_rename.h"
2008-02-27 20:56:47 +03:00
static int _key_cmp(const void *key, const void *data) {
uint64_t a;
csync_file_stat_t *b;
2008-06-27 20:01:19 +04:00
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) {
2008-02-27 20:56:47 +03:00
CSYNC *ctx;
2008-04-28 14:35:29 +04:00
size_t len = 0;
2012-03-02 16:38:39 +04:00
char *home;
int rc;
2008-02-27 20:56:47 +03:00
ctx = c_malloc(sizeof(CSYNC));
if (ctx == NULL) {
return -1;
}
ctx->error_code = CSYNC_ERR_NONE;
2008-04-28 14:35:29 +04:00
/* 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) {
ctx->error_code = CSYNC_ERR_MEM;
free(ctx);
return -1;
}
2008-04-28 14:35:29 +04:00
/* 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) {
ctx->error_code = CSYNC_ERR_MEM;
free(ctx);
2008-02-27 20:56:47 +03:00
return -1;
}
ctx->options.max_depth = MAX_DEPTH;
ctx->options.max_time_difference = MAX_TIME_DIFFERENCE;
2009-01-19 13:32:58 +03:00
ctx->options.unix_extensions = 0;
ctx->options.with_conflict_copys=false;
ctx->options.local_only_mode = false;
2008-02-27 20:56:47 +03:00
ctx->pwd.uid = getuid();
ctx->pwd.euid = geteuid();
2012-03-02 16:38:39 +04:00
home = csync_get_user_home_dir();
if (home == NULL) {
SAFE_FREE(ctx->local.uri);
SAFE_FREE(ctx->remote.uri);
SAFE_FREE(ctx);
errno = ENOMEM;
ctx->error_code = CSYNC_ERR_MEM;
2012-03-02 16:38:39 +04:00
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);
2008-02-27 20:56:47 +03:00
SAFE_FREE(ctx);
errno = ENOMEM;
ctx->error_code = CSYNC_ERR_MEM;
2008-02-27 20:56:47 +03:00
return -1;
}
2013-04-20 14:06:59 +04:00
ctx->local.list = 0;
ctx->remote.list = 0;
ctx->current_fs = NULL;
2013-04-20 14:06:59 +04:00
ctx->abort = false;
2008-02-27 20:56:47 +03:00
*csync = ctx;
return 0;
}
int csync_init(CSYNC *ctx) {
int rc;
time_t timediff = -1;
char *exclude = NULL;
2008-02-27 20:56:47 +03:00
char *lock = NULL;
2008-03-06 19:43:58 +03:00
char *config = NULL;
char errbuf[256] = {0};
if (ctx == NULL) {
2008-02-27 20:56:47 +03:00
errno = EBADF;
return -1;
}
ctx->error_code = CSYNC_ERR_NONE;
2008-02-27 20:56:47 +03:00
/* Do not initialize twice */
if (ctx->status & CSYNC_STATUS_INIT) {
2008-02-27 20:56:47 +03:00
return 1;
}
/* create dir if it doesn't exist */
if (! c_isdir(ctx->options.config_dir)) {
c_mkdirs(ctx->options.config_dir, 0700);
}
/* create lock file */
2013-05-04 18:10:11 +04:00
if (asprintf(&lock, "%s/%s", ctx->local.uri, CSYNC_LOCK_FILE) < 0) {
ctx->error_code = CSYNC_ERR_MEM;
2008-02-27 20:56:47 +03:00
rc = -1;
goto out;
}
2012-10-27 18:05:16 +04:00
if (csync_lock(ctx, lock) < 0) {
ctx->error_code = CSYNC_ERR_LOCK;
2008-02-27 20:56:47 +03:00
rc = -1;
goto out;
}
2008-03-06 19:43:58 +03:00
/* load config file */
if (asprintf(&config, "%s/%s", ctx->options.config_dir, CSYNC_CONF_FILE) < 0) {
ctx->error_code = CSYNC_ERR_MEM;
2008-03-06 19:43:58 +03:00
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);
2008-03-06 19:43:58 +03:00
}
2008-02-27 20:56:47 +03:00
2012-02-22 15:14:34 +04:00
#ifndef _WIN32
/* load global exclude list */
if (asprintf(&exclude, "%s/ocsync/%s", SYSCONFDIR, CSYNC_EXCLUDE_FILE) < 0) {
ctx->error_code = CSYNC_ERR_MEM;
rc = -1;
goto out;
}
if (csync_exclude_load(ctx, exclude) < 0) {
strerror_r(errno, errbuf, sizeof(errbuf));
2012-10-27 18:05:16 +04:00
CSYNC_LOG(CSYNC_LOG_PRIORITY_WARN, "Could not load %s - %s", exclude,
errbuf);
}
SAFE_FREE(exclude);
#endif
/* load user exclude list */
if (asprintf(&exclude, "%s/%s", ctx->options.config_dir, CSYNC_EXCLUDE_FILE) < 0) {
ctx->error_code = 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);
}
2008-07-09 12:10:00 +04:00
/* create/load statedb */
if (! csync_is_statedb_disabled(ctx)) {
rc = asprintf(&ctx->statedb.file, "%s/.csync_journal.db",
ctx->local.uri);
if (rc < 0) {
goto out;
}
CSYNC_LOG(CSYNC_LOG_PRIORITY_DEBUG, "Journal: %s", ctx->statedb.file);
2008-02-27 20:56:47 +03:00
2008-07-09 12:10:00 +04:00
if (csync_statedb_load(ctx, ctx->statedb.file) < 0) {
ctx->error_code = CSYNC_ERR_STATEDB_LOAD;
rc = -1;
goto out;
}
2008-02-27 20:56:47 +03:00
}
2008-04-28 16:08:07 +04:00
ctx->local.type = LOCAL_REPLICA;
2008-04-28 14:40:32 +04:00
/* check for uri */
if ( !ctx->options.local_only_mode && csync_fnmatch("*://*", ctx->remote.uri, 0) == 0) {
2008-04-28 14:40:32 +04:00
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) {
ctx->error_code = CSYNC_ERR_MODULE;
2008-04-28 14:40:32 +04:00
rc = -1;
goto out;
}
/* load module */
retry_vio_init:
2008-04-28 16:08:07 +04:00
rc = csync_vio_init(ctx, module, NULL);
if (rc < 0) {
len = strlen(module);
if (len > 0 && module[len-1] == 's') {
module[len-1] = '\0';
goto retry_vio_init;
}
/* Now vio init finally failed which means a module could not be found. */
ctx->error_code = CSYNC_ERR_MODULE;
CSYNC_LOG(CSYNC_LOG_PRIORITY_FATAL,
"The csync module %s could not be loaded.", module);
SAFE_FREE(module);
2008-04-28 14:40:32 +04:00
goto out;
}
SAFE_FREE(module);
ctx->remote.type = REMOTE_REPLICA;
2008-04-28 14:40:32 +04:00
}
2008-04-28 16:08:07 +04:00
} else {
ctx->remote.type = LOCAL_REPLICA;
}
2008-03-20 12:34:58 +03:00
if(!ctx->options.local_only_mode) {
if(ctx->module.capabilities.time_sync_required) {
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);
ctx->error_code = CSYNC_ERR_TIMESKEW;
rc = -1;
goto out;
} else if (timediff < 0) {
/* error code was set in csync_timediff() */
CSYNC_LOG(CSYNC_LOG_PRIORITY_FATAL, "Synchronisation is not possible!");
/* do not override error code set by timediff */
if(ctx->error_code == CSYNC_ERR_NONE) {
ctx->error_code = CSYNC_ERR_TIMESKEW;
}
rc = -1;
goto out;
}
} else {
CSYNC_LOG(CSYNC_LOG_PRIORITY_TRACE, "Module does not need time synchronization.");
}
if(ctx->module.capabilities.unix_extensions == -1) { /* detect */
if (csync_unix_extensions(ctx) < 0) {
CSYNC_LOG(CSYNC_LOG_PRIORITY_FATAL, "Could not detect filesystem type.");
ctx->error_code = CSYNC_ERR_FILESYSTEM;
rc = -1;
goto out;
}
} else {
/* The module specifies the value for the unix_extensions. */
ctx->options.unix_extensions = ctx->module.capabilities.unix_extensions;
}
}
2012-12-04 20:21:53 +04:00
if (ctx->callbacks.progresscb)
csync_vio_set_property(ctx, "progress_callback", &ctx->callbacks.progresscb);
2012-12-04 20:21:53 +04:00
if (ctx->options.timeout)
csync_vio_set_property(ctx, "timeout", &ctx->options.timeout);
if (c_rbtree_create(&ctx->local.tree, _key_cmp, _data_cmp) < 0) {
ctx->error_code = CSYNC_ERR_TREE;
rc = -1;
goto out;
}
if (c_rbtree_create(&ctx->remote.tree, _key_cmp, _data_cmp) < 0) {
ctx->error_code = CSYNC_ERR_TREE;
rc = -1;
goto out;
}
ctx->status = CSYNC_STATUS_INIT;
2008-02-27 20:56:47 +03:00
2013-05-04 18:10:11 +04:00
csync_lock_remove(ctx, lock);
csync_set_module_property(ctx, "csync_context", ctx);
/* initialize random generator */
srand(time(NULL));
2008-02-27 20:56:47 +03:00
rc = 0;
out:
SAFE_FREE(lock);
SAFE_FREE(exclude);
SAFE_FREE(config);
2008-02-27 20:56:47 +03:00
return rc;
}
int csync_update(CSYNC *ctx) {
int rc = -1;
2008-05-15 21:18:41 +04:00
struct timespec start, finish;
2013-05-04 18:10:11 +04:00
char *lock = NULL;
if (ctx == NULL) {
errno = EBADF;
return -1;
}
ctx->error_code = CSYNC_ERR_NONE;
2013-05-04 18:10:11 +04:00
/* try to create lock file */
if (asprintf(&lock, "%s/%s", ctx->local.uri, CSYNC_LOCK_FILE) < 0) {
ctx->error_code = CSYNC_ERR_MEM;
rc = -1;
return rc;
}
if (csync_lock(ctx, lock) < 0) {
ctx->error_code = CSYNC_ERR_LOCK;
rc = -1;
return rc;
}
2013-05-13 19:45:12 +04:00
SAFE_FREE(lock);
2012-10-27 18:05:16 +04:00
csync_memstat_check(ctx);
/* update detection for local replica */
2012-02-04 16:24:53 +04:00
csync_gettime(&start);
2008-04-28 16:08:07 +04:00
ctx->current = LOCAL_REPLICA;
ctx->replica = ctx->local.type;
rc = csync_ftw(ctx, ctx->local.uri, csync_walker, MAX_DEPTH);
if (rc < 0) {
if(ctx->error_code == CSYNC_ERR_NONE)
ctx->error_code = csync_errno_to_csync_error(CSYNC_ERR_UPDATE);
return -1;
}
2012-02-04 16:24:53 +04:00
csync_gettime(&finish);
2008-05-15 21:18:41 +04:00
CSYNC_LOG(CSYNC_LOG_PRIORITY_DEBUG,
2009-04-22 15:41:46 +04:00
"Update detection for local replica took %.2f seconds walking %zu files.",
c_secdiff(finish, start), c_rbtree_size(ctx->local.tree));
2012-10-27 18:05:16 +04:00
csync_memstat_check(ctx);
/* update detection for remote replica */
if( ! ctx->options.local_only_mode ) {
csync_gettime(&start);
ctx->current = REMOTE_REPLICA;
ctx->replica = ctx->remote.type;
rc = csync_ftw(ctx, ctx->remote.uri, csync_walker, MAX_DEPTH);
if (rc < 0) {
if(ctx->error_code == CSYNC_ERR_NONE )
ctx->error_code = csync_errno_to_csync_error(CSYNC_ERR_UPDATE);
return -1;
}
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));
2012-10-27 18:05:16 +04:00
csync_memstat_check(ctx);
}
ctx->status |= CSYNC_STATUS_UPDATE;
return 0;
}
2008-05-15 15:50:34 +04:00
int csync_reconcile(CSYNC *ctx) {
int rc = -1;
2008-05-15 21:18:41 +04:00
struct timespec start, finish;
2008-05-15 15:50:34 +04:00
if (ctx == NULL) {
errno = EBADF;
return -1;
}
ctx->error_code = CSYNC_ERR_NONE;
2008-05-15 15:50:34 +04:00
/* Reconciliation for local replica */
2012-02-04 16:24:53 +04:00
csync_gettime(&start);
2008-05-15 21:42:03 +04:00
2008-05-15 15:50:34 +04:00
ctx->current = LOCAL_REPLICA;
ctx->replica = ctx->local.type;
rc = csync_reconcile_updates(ctx);
2012-02-04 16:24:53 +04:00
csync_gettime(&finish);
2008-05-15 21:42:03 +04:00
2008-05-15 15:50:34 +04:00
CSYNC_LOG(CSYNC_LOG_PRIORITY_DEBUG,
2009-04-22 15:41:46 +04:00
"Reconciliation for local replica took %.2f seconds visiting %zu files.",
c_secdiff(finish, start), c_rbtree_size(ctx->local.tree));
2008-05-15 15:50:34 +04:00
if (rc < 0) {
if( ctx->error_code == CSYNC_ERR_NONE )
ctx->error_code = csync_errno_to_csync_error( CSYNC_ERR_RECONCILE );
2008-05-15 15:50:34 +04:00
return -1;
}
/* Reconciliation for local replica */
2012-02-04 16:24:53 +04:00
csync_gettime(&start);
2008-05-15 21:42:03 +04:00
ctx->current = REMOTE_REPLICA;
2008-05-15 15:50:34 +04:00
ctx->replica = ctx->remote.type;
rc = csync_reconcile_updates(ctx);
2012-02-04 16:24:53 +04:00
csync_gettime(&finish);
2008-05-15 21:42:03 +04:00
2008-05-15 15:50:34 +04:00
CSYNC_LOG(CSYNC_LOG_PRIORITY_DEBUG,
2009-04-22 15:41:46 +04:00
"Reconciliation for remote replica took %.2f seconds visiting %zu files.",
c_secdiff(finish, start), c_rbtree_size(ctx->remote.tree));
2008-05-15 15:50:34 +04:00
if (rc < 0) {
if( ctx->error_code == CSYNC_ERR_NONE )
ctx->error_code = csync_errno_to_csync_error( CSYNC_ERR_RECONCILE );
2008-05-15 15:50:34 +04:00
return -1;
}
ctx->status |= CSYNC_STATUS_RECONCILE;
2008-05-15 15:50:34 +04:00
return 0;
}
int csync_propagate(CSYNC *ctx) {
int rc = -1;
struct timespec start, finish;
if (ctx == NULL) {
errno = EBADF;
return -1;
}
ctx->error_code = CSYNC_ERR_NONE;
2013-01-04 23:45:10 +04:00
ctx->current = REMOTE_REPLICA;
ctx->replica = ctx->remote.type;
rc = csync_propagate_renames(ctx);
2013-01-04 23:45:10 +04:00
if (rc < 0) {
if( ctx->error_code == CSYNC_ERR_NONE )
ctx->error_code = csync_errno_to_csync_error( CSYNC_ERR_PROPAGATE);
return -1;
}
/* Reconciliation for local replica */
2012-02-04 16:24:53 +04:00
csync_gettime(&start);
ctx->current = LOCAL_REPLICA;
ctx->replica = ctx->local.type;
2008-12-14 01:29:16 +03:00
rc = csync_propagate_files(ctx);
2012-02-04 16:24:53 +04:00
csync_gettime(&finish);
CSYNC_LOG(CSYNC_LOG_PRIORITY_DEBUG,
2009-04-22 15:41:46 +04:00
"Propagation for local replica took %.2f seconds visiting %zu files.",
c_secdiff(finish, start), c_rbtree_size(ctx->local.tree));
if (rc < 0) {
if( ctx->error_code == CSYNC_ERR_NONE )
ctx->error_code = csync_errno_to_csync_error( CSYNC_ERR_PROPAGATE);
return -1;
}
/* Reconciliation for local replica */
2012-02-04 16:24:53 +04:00
csync_gettime(&start);
ctx->current = REMOTE_REPLICA;
ctx->replica = ctx->remote.type;
2008-12-14 01:29:16 +03:00
rc = csync_propagate_files(ctx);
2012-02-04 16:24:53 +04:00
csync_gettime(&finish);
CSYNC_LOG(CSYNC_LOG_PRIORITY_DEBUG,
2009-04-22 15:41:46 +04:00
"Propagation for remote replica took %.2f seconds visiting %zu files.",
c_secdiff(finish, start), c_rbtree_size(ctx->remote.tree));
if (rc < 0) {
if( ctx->error_code == CSYNC_ERR_NONE )
ctx->error_code = csync_errno_to_csync_error( 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) {
int rc = 0;
csync_file_stat_t *cur = NULL;
CSYNC *ctx = NULL;
c_rbtree_visit_func *visitor = NULL;
_csync_treewalk_context *twctx = NULL;
TREE_WALK_FILE trav;
cur = (csync_file_stat_t *) obj;
ctx = (CSYNC *) data;
if (ctx == NULL) {
2013-04-05 17:05:44 +04:00
return -1;
}
if (obj == NULL || data == NULL) {
ctx->error_code = CSYNC_ERR_PARAM;
return -1;
}
ctx->error_code = CSYNC_ERR_NONE;
twctx = (_csync_treewalk_context*) ctx->callbacks.userdata;
if (twctx == NULL) {
ctx->error_code = CSYNC_ERR_PARAM;
return -1;
}
if (twctx->instruction_filter > 0 &&
!(twctx->instruction_filter & cur->instruction) ) {
return 0;
}
visitor = (c_rbtree_visit_func*)(twctx->user_visitor);
if (visitor != NULL) {
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;
trav.rename_path = cur->destpath;
trav.md5 = cur->md5;
trav.error_string = cur->error_string;
rc = (*visitor)(&trav, twctx->userdata);
cur->instruction = trav.instruction;
if (trav.md5 != cur->md5) {
SAFE_FREE(cur->md5);
cur->md5 = c_strdup(trav.md5);
}
return rc;
}
ctx->error_code = 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;
if (ctx == NULL) {
errno = EBADF;
return -1;
}
ctx->error_code = CSYNC_ERR_NONE;
if(!(visitor && tree)) {
ctx->error_code = CSYNC_ERR_PARAM;
return rc;
}
tw_ctx.userdata = ctx->callbacks.userdata;
tw_ctx.user_visitor = visitor;
tw_ctx.instruction_filter = filter;
ctx->callbacks.userdata = &tw_ctx;
rc = c_rbtree_walk(tree, (void*) ctx, _csync_treewalk_visitor);
if( rc < 0 ) {
if( ctx->error_code == CSYNC_ERR_NONE )
ctx->error_code = csync_errno_to_csync_error(CSYNC_ERR_TREE);
}
ctx->callbacks.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;
if(ctx) {
tree = ctx->remote.tree;
}
/* all error handling in the called function */
rc = _csync_walk_tree(ctx, tree, visitor, filter);
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;
if(ctx) {
tree = ctx->local.tree;
}
/* all error handling in the called function */
rc = _csync_walk_tree(ctx, tree, visitor, filter);
return rc;
}
static void _tree_destructor(void *data) {
csync_file_stat_t *freedata = NULL;
freedata = (csync_file_stat_t *) data;
csync_file_stat_free(freedata);
}
static int _merge_and_write_statedb(CSYNC *ctx) {
2008-05-15 21:18:41 +04:00
struct timespec start, finish;
char errbuf[256] = {0};
2008-04-29 13:22:06 +04:00
int jwritten = 0;
int rc = 0;
2008-07-09 12:10:00 +04:00
/* if we have a statedb */
if (ctx->statedb.db != NULL) {
2008-06-18 12:41:49 +04:00
/* 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);
rc = -1;
} else {
2012-02-04 16:24:53 +04:00
csync_gettime(&start);
2008-07-09 12:10:00 +04:00
/* write the statedb to disk */
if (csync_statedb_write(ctx) == 0) {
jwritten = 1;
2012-02-04 16:24:53 +04:00
csync_gettime(&finish);
CSYNC_LOG(CSYNC_LOG_PRIORITY_DEBUG,
2008-07-09 12:10:00 +04:00
"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));
2008-07-09 12:10:00 +04:00
CSYNC_LOG(CSYNC_LOG_PRIORITY_ERROR, "Unable to write statedb: %s",
errbuf);
rc = -1;
}
2008-04-29 13:22:06 +04:00
}
}
if (csync_statedb_close(ctx, ctx->statedb.file, jwritten) < 0) {
CSYNC_LOG(CSYNC_LOG_PRIORITY_DEBUG, "ERR: closing of statedb failed.");
rc = -1;
}
}
return rc;
}
int csync_commit(CSYNC *ctx) {
int rc = 0;
2013-05-04 18:10:11 +04:00
char *lock = NULL;
/* maybe the propagate was done using another propagator, let the merger think it has been done */
2013-05-10 15:11:01 +04:00
if (ctx->error_code == CSYNC_ERR_NONE)
ctx->status = CSYNC_STATUS_DONE;
ctx->error_code = CSYNC_ERR_NONE;
if (_merge_and_write_statedb(ctx) < 0) {
CSYNC_LOG(CSYNC_LOG_PRIORITY_DEBUG, "Merge and Write database failed!");
if (ctx->error_code == CSYNC_ERR_NONE) {
ctx->error_code = CSYNC_ERR_STATEDB_WRITE;
}
rc = 1; /* Set to soft error. */
/* The other steps happen anyway, what else can we do? */
}
2013-04-22 15:51:44 +04:00
csync_vio_commit(ctx);
while (ctx->progress) {
csync_progressinfo_t *next = ctx->progress->next;
csync_statedb_free_progressinfo(ctx->progress);
ctx->progress = next;
}
/* 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);
}
csync_rename_destroy(ctx);
/* 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);
ctx->remote.list = 0;
ctx->local.list = 0;
ctx->remote.read_from_db = 0;
/* create/load statedb */
if (! csync_is_statedb_disabled(ctx)) {
if(!ctx->statedb.file) {
rc = asprintf(&ctx->statedb.file, "%s/.csync_journal.db",
ctx->local.uri);
if (rc < 0) {
ctx->error_code = CSYNC_ERR_MEM;
CSYNC_LOG(CSYNC_LOG_PRIORITY_DEBUG, "Failed to assemble statedb file name.");
goto out;
}
}
CSYNC_LOG(CSYNC_LOG_PRIORITY_DEBUG, "Journal: %s", ctx->statedb.file);
if (csync_statedb_load(ctx, ctx->statedb.file) < 0) {
ctx->error_code = CSYNC_ERR_STATEDB_LOAD;
rc = -1;
goto out;
}
}
/* Create new trees */
if (c_rbtree_create(&ctx->local.tree, _key_cmp, _data_cmp) < 0) {
ctx->error_code = CSYNC_ERR_TREE;
rc = -1;
goto out;
}
if (c_rbtree_create(&ctx->remote.tree, _key_cmp, _data_cmp) < 0) {
ctx->error_code = CSYNC_ERR_TREE;
rc = -1;
goto out;
}
ctx->status = CSYNC_STATUS_INIT;
ctx->error_code = CSYNC_ERR_NONE;
SAFE_FREE(ctx->error_string);
2013-05-04 18:10:11 +04:00
/* create lock file */
if (asprintf(&lock, "%s/%s", ctx->local.uri, CSYNC_LOCK_FILE) < 0) {
ctx->error_code = CSYNC_ERR_MEM;
rc = -1;
goto out;
}
csync_lock_remove(ctx, lock);
2013-05-13 19:45:12 +04:00
SAFE_FREE(lock);
2013-05-04 18:10:11 +04:00
out:
return rc;
}
int csync_destroy(CSYNC *ctx) {
char *lock = NULL;
if (ctx == NULL) {
errno = EBADF;
return -1;
}
ctx->error_code = CSYNC_ERR_NONE;
csync_vio_shutdown(ctx);
if (_merge_and_write_statedb(ctx) < 0) {
CSYNC_LOG(CSYNC_LOG_PRIORITY_DEBUG, "destroy: Merge and Write database failed!");
if (ctx->error_code == CSYNC_ERR_NONE) {
ctx->error_code = CSYNC_ERR_STATEDB_WRITE;
}
/* The other steps happen anyway, what else can we do? */
2008-02-27 20:56:47 +03:00
}
2008-06-18 12:41:49 +04:00
/* clear exclude list */
csync_exclude_destroy(ctx);
2008-06-18 12:41:49 +04:00
/* remove the lock file */
2008-02-27 20:56:47 +03:00
if (asprintf(&lock, "%s/%s", ctx->options.config_dir, CSYNC_LOCK_FILE) > 0) {
2012-10-27 18:05:16 +04:00
csync_lock_remove(ctx, lock);
2008-02-27 20:56:47 +03:00
}
while (ctx->progress) {
csync_progressinfo_t *next = ctx->progress->next;
csync_statedb_free_progressinfo(ctx->progress);
ctx->progress = next;
}
/* 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);
}
2013-01-04 23:45:10 +04:00
csync_rename_destroy(ctx);
2008-06-18 12:41:49 +04:00
/* 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);
2008-02-27 20:56:47 +03:00
SAFE_FREE(ctx->options.config_dir);
2008-07-09 12:10:00 +04:00
SAFE_FREE(ctx->statedb.file);
SAFE_FREE(ctx->error_string);
2008-02-27 20:56:47 +03:00
#ifdef WITH_ICONV
c_close_iconv();
#endif
2008-02-27 20:56:47 +03:00
SAFE_FREE(ctx);
SAFE_FREE(lock);
return 0;
}
2009-03-26 12:40:16 +03:00
/* Check if csync is the required version or get the version string. */
const char *csync_version(int req_version) {
if (req_version <= LIBCSYNC_VERSION_INT) {
return CSYNC_STRINGIFY(LIBCSYNC_VERSION);
}
return NULL;
2008-02-27 20:56:47 +03:00
}
int csync_add_exclude_list(CSYNC *ctx, const char *path) {
2008-05-20 18:31:10 +04:00
if (ctx == NULL || path == NULL) {
return -1;
}
ctx->error_code = CSYNC_ERR_NONE;
2008-05-20 18:31:10 +04:00
return csync_exclude_load(ctx, path);
}
2009-03-26 13:24:34 +03:00
const char *csync_get_config_dir(CSYNC *ctx) {
if (ctx == NULL) {
return NULL;
}
ctx->error_code = CSYNC_ERR_NONE;
return ctx->options.config_dir;
}
int csync_set_config_dir(CSYNC *ctx, const char *path) {
if (ctx == NULL || path == NULL) {
return -1;
}
ctx->error_code = CSYNC_ERR_NONE;
SAFE_FREE(ctx->options.config_dir);
ctx->options.config_dir = c_strdup(path);
if (ctx->options.config_dir == NULL) {
ctx->error_code = CSYNC_ERR_MEM;
return -1;
}
return 0;
}
2008-07-09 12:10:00 +04:00
int csync_enable_statedb(CSYNC *ctx) {
if (ctx == NULL) {
return -1;
}
ctx->error_code = CSYNC_ERR_NONE;
2008-06-24 19:39:46 +04:00
if (ctx->status & CSYNC_STATUS_INIT) {
ctx->error_code = CSYNC_ERR_UNSPEC;
fprintf(stderr, "csync_enable_statedb: This function must be called before initialization.\n");
2008-06-24 19:39:46 +04:00
return -1;
}
2008-07-09 12:10:00 +04:00
ctx->statedb.disabled = 0;
return 0;
}
2008-07-09 12:10:00 +04:00
int csync_disable_statedb(CSYNC *ctx) {
if (ctx == NULL) {
return -1;
}
ctx->error_code = CSYNC_ERR_NONE;
2008-06-24 19:39:46 +04:00
if (ctx->status & CSYNC_STATUS_INIT) {
ctx->error_code = CSYNC_ERR_UNSPEC;
fprintf(stderr, "csync_disable_statedb: This function must be called before initialization.\n");
2008-06-24 19:39:46 +04:00
return -1;
}
2008-07-09 12:10:00 +04:00
ctx->statedb.disabled = 1;
return 0;
}
2008-07-09 12:10:00 +04:00
int csync_is_statedb_disabled(CSYNC *ctx) {
if (ctx == NULL) {
2008-05-21 16:46:42 +04:00
return -1;
}
ctx->error_code = CSYNC_ERR_NONE;
2008-05-21 16:46:42 +04:00
2008-07-09 12:10:00 +04:00
return ctx->statedb.disabled;
}
2008-06-24 15:36:27 +04:00
int csync_set_auth_callback(CSYNC *ctx, csync_auth_callback cb) {
if (ctx == NULL || cb == NULL) {
return -1;
}
ctx->error_code = CSYNC_ERR_NONE;
if (ctx->status & CSYNC_STATUS_INIT) {
fprintf(stderr, "csync_set_auth_callback: This function must be called before initialization.\n");
ctx->error_code = CSYNC_ERR_UNSPEC;
return -1;
}
ctx->callbacks.auth_function = cb;
return 0;
}
int csync_set_log_verbosity(CSYNC *ctx, int verbosity) {
if (ctx == NULL || verbosity < 0) {
return -1;
}
ctx->options.log_verbosity = verbosity;
return 0;
}
int csync_get_log_verbosity(CSYNC *ctx) {
if (ctx == NULL) {
return -1;
}
return ctx->options.log_verbosity;
}
2012-10-27 17:15:59 +04:00
int csync_set_log_callback(CSYNC *ctx, csync_log_callback cb) {
if (ctx == NULL || cb == NULL) {
return -1;
}
ctx->callbacks.log_function = cb;
return 0;
}
2009-03-26 13:24:34 +03:00
const char *csync_get_statedb_file(CSYNC *ctx) {
if (ctx == NULL) {
return NULL;
}
ctx->error_code = CSYNC_ERR_NONE;
2008-07-09 12:10:00 +04:00
return c_strdup(ctx->statedb.file);
}
void *csync_get_userdata(CSYNC *ctx) {
if (ctx == NULL) {
return NULL;
}
ctx->error_code = CSYNC_ERR_NONE;
return ctx->callbacks.userdata;
}
int csync_set_userdata(CSYNC *ctx, void *userdata) {
if (ctx == NULL) {
return -1;
}
ctx->error_code = CSYNC_ERR_NONE;
ctx->callbacks.userdata = userdata;
return 0;
}
2008-06-24 15:36:27 +04:00
csync_auth_callback csync_get_auth_callback(CSYNC *ctx) {
if (ctx == NULL) {
return NULL;
}
ctx->error_code = CSYNC_ERR_NONE;
return ctx->callbacks.auth_function;
2012-10-27 17:15:59 +04:00
}
csync_log_callback csync_get_log_callback(CSYNC *ctx) {
if (ctx == NULL) {
return NULL;
}
return ctx->callbacks.log_function;
}
int csync_set_status(CSYNC *ctx, int status) {
if (ctx == NULL || status < 0) {
return -1;
}
ctx->error_code = CSYNC_ERR_NONE;
ctx->status = status;
return 0;
}
int csync_get_status(CSYNC *ctx) {
2008-06-24 13:13:56 +04:00
if (ctx == NULL) {
return -1;
}
ctx->error_code = CSYNC_ERR_NONE;
2008-06-24 13:13:56 +04:00
return ctx->status;
}
int csync_enable_conflictcopys(CSYNC* ctx){
if (ctx == NULL) {
return -1;
}
ctx->error_code = CSYNC_ERR_NONE;
if (ctx->status & CSYNC_STATUS_INIT) {
fprintf(stderr, "csync_enable_conflictcopys: This function must be called before initialization.\n");
ctx->error_code = CSYNC_ERR_UNSPEC;
return -1;
}
ctx->options.with_conflict_copys=true;
return 0;
}
int csync_set_local_only(CSYNC *ctx, bool local_only) {
if (ctx == NULL) {
return -1;
}
ctx->error_code = CSYNC_ERR_NONE;
if (ctx->status & CSYNC_STATUS_INIT) {
fprintf(stderr, "csync_set_local_only: This function must be called before initialization.\n");
ctx->error_code = CSYNC_ERR_UNSPEC;
return -1;
}
ctx->options.local_only_mode=local_only;
return 0;
}
bool csync_get_local_only(CSYNC *ctx) {
if (ctx == NULL) {
return -1;
}
ctx->error_code = CSYNC_ERR_NONE;
return ctx->options.local_only_mode;
}
CSYNC_ERROR_CODE csync_get_error(CSYNC *ctx) {
if (ctx == NULL) {
return CSYNC_ERR_PARAM;
}
return ctx->error_code;
}
const char *csync_get_error_string(CSYNC *ctx)
{
return csync_vio_get_error_string(ctx);
}
int csync_set_module_property(CSYNC* ctx, const char* key, void* value)
{
if (!(ctx->status & CSYNC_STATUS_INIT)) {
ctx->error_code = CSYNC_ERR_UNSPEC;
fprintf(stderr, "csync_set_module_property: This function must be called after initialization.\n");
return -1;
}
return csync_vio_set_property(ctx, key, value);
}
#ifdef WITH_ICONV
int csync_set_iconv_codec(const char *from)
{
c_close_iconv();
if (from != NULL) {
c_setup_iconv(from);
}
return 0;
}
#endif
csync_progress_callback csync_get_progress_callback(CSYNC *ctx)
{
if ( ctx==NULL ) {
return NULL;
}
return ctx->callbacks.progresscb;
}
2012-12-04 20:21:53 +04:00
int csync_set_progress_callback(CSYNC* ctx, csync_progress_callback cb)
{
2013-04-05 17:08:28 +04:00
if (ctx == NULL) {
return -1;
}
if (cb == NULL ) {
2012-12-04 20:21:53 +04:00
ctx->error_code = CSYNC_ERR_PARAM;
return -1;
}
2013-04-05 17:08:28 +04:00
2012-12-04 20:21:53 +04:00
ctx->error_code = CSYNC_ERR_NONE;
ctx->callbacks.progresscb = cb;
return 0;
}
void csync_request_abort(CSYNC *ctx)
{
2013-05-16 19:37:30 +04:00
if (ctx != NULL) {
ctx->abort = true;
2013-05-16 19:37:30 +04:00
}
}
void csync_resume(CSYNC *ctx)
{
2013-05-16 19:37:30 +04:00
if (ctx != NULL) {
ctx->abort = false;
2013-05-16 19:37:30 +04:00
}
}
int csync_abort_requested(CSYNC *ctx)
{
if (ctx != NULL) {
return ctx->abort;
} else {
return (1 == 0);
}
}
void csync_file_stat_free(csync_file_stat_t *st)
{
if (st) {
SAFE_FREE(st->md5);
SAFE_FREE(st->error_string);
SAFE_FREE(st->destpath);
SAFE_FREE(st);
}
}
2009-05-13 12:12:07 +04:00
/* vim: set ts=8 sw=2 et cindent: */