2008-02-27 20:56:47 +03:00
|
|
|
/*
|
|
|
|
* libcsync -- a library to sync a directory with another
|
|
|
|
*
|
2013-07-23 19:31:55 +04:00
|
|
|
* Copyright (c) 2008-2013 by Andreas Schneider <asn@cryptomilk.org>
|
|
|
|
* Copyright (c) 2012-2013 by Klaas Freitag <freitag@owncloud.com>
|
2008-02-27 20:56:47 +03:00
|
|
|
*
|
2013-07-23 19:31:55 +04:00
|
|
|
* This library is free software; you can redistribute it and/or
|
|
|
|
* modify it under the terms of the GNU Lesser General Public
|
|
|
|
* License as published by the Free Software Foundation; either
|
|
|
|
* version 2.1 of the License, or (at your option) any later version.
|
2008-02-27 20:56:47 +03:00
|
|
|
*
|
2013-07-23 19:31:55 +04:00
|
|
|
* This library is distributed in the hope that it will be useful,
|
2008-07-10 12:25:12 +04:00
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
2013-07-23 19:31:55 +04:00
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
|
|
* Lesser General Public License for more details.
|
2008-02-27 20:56:47 +03:00
|
|
|
*
|
2013-07-23 19:31:55 +04:00
|
|
|
* You should have received a copy of the GNU Lesser General Public
|
|
|
|
* License along with this library; 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
|
|
|
*/
|
|
|
|
|
2014-01-15 15:20:03 +04:00
|
|
|
#include "config_csync.h"
|
2012-03-02 19:47:34 +04:00
|
|
|
|
2008-06-09 19:19:12 +04:00
|
|
|
#ifndef _GNU_SOURCE
|
|
|
|
#define _GNU_SOURCE
|
|
|
|
#endif
|
2008-04-28 14:40:32 +04:00
|
|
|
|
2016-02-10 21:24:40 +03:00
|
|
|
#include <assert.h>
|
2008-02-27 20:56:47 +03:00
|
|
|
#include <errno.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <string.h>
|
2008-04-26 12:43:22 +04:00
|
|
|
#include <time.h>
|
2008-11-13 16:08:26 +03:00
|
|
|
#include <sys/types.h>
|
2011-04-06 18:43:04 +04:00
|
|
|
#include <stdbool.h>
|
2008-02-27 20:56:47 +03:00
|
|
|
|
|
|
|
#include "c_lib.h"
|
|
|
|
#include "csync_private.h"
|
2008-03-25 18:22:51 +03:00
|
|
|
#include "csync_exclude.h"
|
2008-07-09 11:57:19 +04:00
|
|
|
#include "csync_statedb.h"
|
2008-05-05 12:48:05 +04:00
|
|
|
#include "csync_time.h"
|
2008-04-28 18:49:21 +04:00
|
|
|
#include "csync_util.h"
|
2012-03-02 16:38:39 +04:00
|
|
|
#include "csync_misc.h"
|
2012-12-07 16:02:46 +04:00
|
|
|
#include "std/c_private.h"
|
2013-04-26 12:45:14 +04:00
|
|
|
|
2008-04-23 14:12:48 +04:00
|
|
|
#include "csync_update.h"
|
2008-05-15 15:50:34 +04:00
|
|
|
#include "csync_reconcile.h"
|
2008-04-23 14:12:48 +04:00
|
|
|
|
2008-04-22 17:58:06 +04:00
|
|
|
#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"
|
2017-09-01 19:11:43 +03:00
|
|
|
#include "common/c_jhash.h"
|
2014-03-24 14:35:19 +04:00
|
|
|
|
2008-04-28 15:42:10 +04:00
|
|
|
|
2017-09-04 20:06:13 +03:00
|
|
|
csync_s::csync_s(const char *localUri, const char *db_file) {
|
2008-04-28 14:35:29 +04:00
|
|
|
size_t len = 0;
|
2008-02-27 20:56:47 +03:00
|
|
|
|
2008-04-28 14:35:29 +04:00
|
|
|
/* remove trailing slashes */
|
2017-09-04 20:06:13 +03:00
|
|
|
len = strlen(localUri);
|
|
|
|
while(len > 0 && localUri[len - 1] == '/') --len;
|
2008-02-27 20:56:47 +03:00
|
|
|
|
2017-09-04 20:06:13 +03:00
|
|
|
local.uri = c_strndup(localUri, len);
|
2013-04-20 14:06:59 +04:00
|
|
|
|
2017-09-04 20:06:13 +03:00
|
|
|
statedb.file = c_strdup(db_file);
|
2008-02-27 20:56:47 +03:00
|
|
|
}
|
|
|
|
|
2008-04-23 14:12:48 +04:00
|
|
|
int csync_update(CSYNC *ctx) {
|
2008-05-15 14:03:05 +04:00
|
|
|
int rc = -1;
|
2008-05-15 21:18:41 +04:00
|
|
|
struct timespec start, finish;
|
2013-05-04 18:10:11 +04:00
|
|
|
|
2008-04-26 12:43:22 +04:00
|
|
|
if (ctx == NULL) {
|
|
|
|
errno = EBADF;
|
|
|
|
return -1;
|
|
|
|
}
|
2013-08-18 19:26:45 +04:00
|
|
|
ctx->status_code = CSYNC_STATUS_OK;
|
2008-04-26 12:43:22 +04:00
|
|
|
|
2016-07-10 13:54:33 +03:00
|
|
|
/* Path of database file is set in csync_init */
|
|
|
|
if (csync_statedb_load(ctx, ctx->statedb.file, &ctx->statedb.db) < 0) {
|
2013-08-05 16:58:43 +04:00
|
|
|
rc = -1;
|
|
|
|
return rc;
|
2016-07-10 13:54:33 +03:00
|
|
|
}
|
2013-08-05 16:58:43 +04:00
|
|
|
|
2013-03-12 20:01:39 +04:00
|
|
|
ctx->status_code = CSYNC_STATUS_OK;
|
|
|
|
|
2013-03-01 12:59:55 +04:00
|
|
|
csync_memstat_check();
|
2008-04-28 18:49:21 +04:00
|
|
|
|
2014-03-27 20:11:19 +04:00
|
|
|
if (!ctx->excludes) {
|
2017-03-31 17:13:01 +03:00
|
|
|
CSYNC_LOG(CSYNC_LOG_PRIORITY_INFO, "No exclude file loaded or defined!");
|
2014-03-27 20:11:19 +04:00
|
|
|
}
|
|
|
|
|
2008-05-15 14:03:05 +04:00
|
|
|
/* 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;
|
2008-05-15 14:03:05 +04:00
|
|
|
|
|
|
|
rc = csync_ftw(ctx, ctx->local.uri, csync_walker, MAX_DEPTH);
|
2013-06-06 11:34:54 +04:00
|
|
|
if (rc < 0) {
|
2014-10-22 17:41:29 +04:00
|
|
|
if(ctx->status_code == CSYNC_STATUS_OK) {
|
2013-08-18 19:26:45 +04:00
|
|
|
ctx->status_code = csync_errno_to_status(errno, CSYNC_STATUS_UPDATE_ERROR);
|
2014-10-22 17:41:29 +04:00
|
|
|
}
|
|
|
|
goto out;
|
2013-06-06 11:34:54 +04:00
|
|
|
}
|
2008-05-15 14:03:05 +04:00
|
|
|
|
2012-02-04 16:24:53 +04:00
|
|
|
csync_gettime(&finish);
|
2008-05-15 21:18:41 +04:00
|
|
|
|
2008-04-26 12:43:22 +04:00
|
|
|
CSYNC_LOG(CSYNC_LOG_PRIORITY_DEBUG,
|
2013-03-12 20:01:39 +04:00
|
|
|
"Update detection for local replica took %.2f seconds walking %zu files.",
|
2017-08-23 20:16:12 +03:00
|
|
|
c_secdiff(finish, start), ctx->local.files.size());
|
2013-03-01 12:59:55 +04:00
|
|
|
csync_memstat_check();
|
2008-04-26 12:43:22 +04:00
|
|
|
|
2008-05-15 14:03:05 +04:00
|
|
|
/* update detection for remote replica */
|
2014-06-03 19:52:07 +04:00
|
|
|
csync_gettime(&start);
|
|
|
|
ctx->current = REMOTE_REPLICA;
|
2008-05-15 14:03:05 +04:00
|
|
|
|
2016-11-15 20:47:04 +03:00
|
|
|
rc = csync_ftw(ctx, "", csync_walker, MAX_DEPTH);
|
2014-06-03 19:52:07 +04:00
|
|
|
if (rc < 0) {
|
2014-10-22 17:41:29 +04:00
|
|
|
if(ctx->status_code == CSYNC_STATUS_OK) {
|
2014-06-03 19:52:07 +04:00
|
|
|
ctx->status_code = csync_errno_to_status(errno, CSYNC_STATUS_UPDATE_ERROR);
|
2014-10-22 17:41:29 +04:00
|
|
|
}
|
|
|
|
goto out;
|
2014-06-03 19:52:07 +04:00
|
|
|
}
|
2008-05-15 14:03:05 +04:00
|
|
|
|
2014-06-03 19:52:07 +04:00
|
|
|
csync_gettime(&finish);
|
2008-04-23 14:12:48 +04:00
|
|
|
|
2014-06-03 19:52:07 +04:00
|
|
|
CSYNC_LOG(CSYNC_LOG_PRIORITY_DEBUG,
|
|
|
|
"Update detection for remote replica took %.2f seconds "
|
|
|
|
"walking %zu files.",
|
2017-08-23 20:16:12 +03:00
|
|
|
c_secdiff(finish, start), ctx->remote.files.size());
|
2014-06-03 19:52:07 +04:00
|
|
|
csync_memstat_check();
|
2008-05-15 14:03:05 +04:00
|
|
|
|
2008-05-27 16:15:44 +04:00
|
|
|
ctx->status |= CSYNC_STATUS_UPDATE;
|
2008-04-29 13:20:52 +04:00
|
|
|
|
2014-10-22 17:41:29 +04:00
|
|
|
rc = 0;
|
|
|
|
|
|
|
|
out:
|
|
|
|
csync_statedb_close(ctx);
|
|
|
|
return rc;
|
2008-04-23 14:12:48 +04:00
|
|
|
}
|
|
|
|
|
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;
|
|
|
|
}
|
2013-03-12 20:01:39 +04:00
|
|
|
ctx->status_code = CSYNC_STATUS_OK;
|
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
|
|
|
|
2014-10-22 17:41:29 +04:00
|
|
|
if (csync_statedb_load(ctx, ctx->statedb.file, &ctx->statedb.db) < 0) {
|
|
|
|
rc = -1;
|
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
|
2008-05-15 15:50:34 +04:00
|
|
|
ctx->current = LOCAL_REPLICA;
|
|
|
|
|
|
|
|
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.",
|
2017-08-23 20:16:12 +03:00
|
|
|
c_secdiff(finish, start), ctx->local.files.size());
|
2008-05-15 15:50:34 +04:00
|
|
|
|
|
|
|
if (rc < 0) {
|
2013-04-06 20:47:08 +04:00
|
|
|
if (!CSYNC_STATUS_IS_OK(ctx->status_code)) {
|
2013-08-18 19:26:45 +04:00
|
|
|
ctx->status_code = csync_errno_to_status( errno, CSYNC_STATUS_RECONCILE_ERROR );
|
2013-04-06 20:47:08 +04:00
|
|
|
}
|
2014-10-22 17:41:29 +04:00
|
|
|
goto out;
|
2008-05-15 15:50:34 +04:00
|
|
|
}
|
|
|
|
|
2014-03-24 14:35:19 +04:00
|
|
|
/* Reconciliation for remote replica */
|
2012-02-04 16:24:53 +04:00
|
|
|
csync_gettime(&start);
|
2008-05-15 21:42:03 +04:00
|
|
|
|
2012-10-19 16:23:20 +04:00
|
|
|
ctx->current = REMOTE_REPLICA;
|
2008-05-15 15:50:34 +04:00
|
|
|
|
|
|
|
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.",
|
2017-08-23 20:16:12 +03:00
|
|
|
c_secdiff(finish, start), ctx->remote.files.size());
|
2008-05-15 15:50:34 +04:00
|
|
|
|
|
|
|
if (rc < 0) {
|
2013-04-06 20:47:08 +04:00
|
|
|
if (!CSYNC_STATUS_IS_OK(ctx->status_code)) {
|
2013-08-18 19:26:45 +04:00
|
|
|
ctx->status_code = csync_errno_to_status(errno, CSYNC_STATUS_RECONCILE_ERROR );
|
2013-04-06 20:47:08 +04:00
|
|
|
}
|
2014-10-22 17:41:29 +04:00
|
|
|
goto out;
|
2008-05-15 15:50:34 +04:00
|
|
|
}
|
|
|
|
|
2008-05-27 16:15:44 +04:00
|
|
|
ctx->status |= CSYNC_STATUS_RECONCILE;
|
2008-05-15 15:50:34 +04:00
|
|
|
|
2014-10-22 17:41:29 +04:00
|
|
|
rc = 0;
|
|
|
|
|
|
|
|
out:
|
|
|
|
csync_statedb_close(ctx);
|
2008-05-15 15:50:34 +04:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2012-03-19 19:05:23 +04:00
|
|
|
/*
|
|
|
|
* local visitor which calls the user visitor with repacked stat info.
|
|
|
|
*/
|
2017-08-23 20:16:12 +03:00
|
|
|
static int _csync_treewalk_visitor(csync_file_stat_t *cur, CSYNC * ctx) {
|
2012-12-15 19:34:31 +04:00
|
|
|
int rc = 0;
|
2017-08-17 15:39:23 +03:00
|
|
|
csync_treewalk_visit_func *visitor = NULL;
|
2012-03-19 19:05:23 +04:00
|
|
|
_csync_treewalk_context *twctx = NULL;
|
2017-08-23 20:16:12 +03:00
|
|
|
csync_s::FileMap *other_tree = nullptr;
|
2012-10-28 14:31:25 +04:00
|
|
|
|
2013-04-08 11:11:25 +04:00
|
|
|
if (ctx == NULL) {
|
2013-04-05 17:05:44 +04:00
|
|
|
return -1;
|
|
|
|
}
|
2012-12-03 20:32:08 +04:00
|
|
|
|
2014-03-24 14:35:19 +04:00
|
|
|
/* we need the opposite tree! */
|
|
|
|
switch (ctx->current) {
|
|
|
|
case LOCAL_REPLICA:
|
2017-08-23 20:16:12 +03:00
|
|
|
other_tree = &ctx->remote.files;
|
2014-03-24 14:35:19 +04:00
|
|
|
break;
|
|
|
|
case REMOTE_REPLICA:
|
2017-08-23 20:16:12 +03:00
|
|
|
other_tree = &ctx->local.files;
|
2014-03-24 14:35:19 +04:00
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2017-08-23 20:16:12 +03:00
|
|
|
csync_s::FileMap::const_iterator other_file_it = other_tree->find(cur->path);
|
2014-03-24 14:35:19 +04:00
|
|
|
|
2017-08-23 20:16:12 +03:00
|
|
|
if (other_file_it == other_tree->cend()) {
|
2014-03-24 14:35:19 +04:00
|
|
|
/* Check the renamed path as well. */
|
2017-08-23 20:16:12 +03:00
|
|
|
QByteArray renamed_path = csync_rename_adjust_path(ctx, cur->path);
|
|
|
|
if (renamed_path != cur->path)
|
|
|
|
other_file_it = other_tree->find(renamed_path);
|
2014-03-24 14:35:19 +04:00
|
|
|
}
|
|
|
|
|
2017-08-23 20:16:12 +03:00
|
|
|
if (other_file_it == other_tree->cend()) {
|
2015-05-12 17:32:00 +03:00
|
|
|
/* Check the source path as well. */
|
2017-08-23 20:16:12 +03:00
|
|
|
QByteArray renamed_path = csync_rename_adjust_path_source(ctx, cur->path);
|
|
|
|
if (renamed_path != cur->path)
|
|
|
|
other_file_it = other_tree->find(renamed_path);
|
2015-05-12 17:32:00 +03:00
|
|
|
}
|
|
|
|
|
2017-08-23 20:16:12 +03:00
|
|
|
csync_file_stat_t *other = (other_file_it != other_tree->cend()) ? other_file_it->second.get() : NULL;
|
2017-08-17 15:39:23 +03:00
|
|
|
|
2013-08-18 19:26:45 +04:00
|
|
|
ctx->status_code = CSYNC_STATUS_OK;
|
2012-04-17 14:31:27 +04:00
|
|
|
|
2012-10-27 17:10:16 +04:00
|
|
|
twctx = (_csync_treewalk_context*) ctx->callbacks.userdata;
|
2012-10-19 21:20:45 +04:00
|
|
|
if (twctx == NULL) {
|
2013-03-12 20:01:39 +04:00
|
|
|
ctx->status_code = CSYNC_STATUS_PARAM_ERROR;
|
2012-10-19 21:20:45 +04:00
|
|
|
return -1;
|
|
|
|
}
|
2012-04-17 14:31:27 +04:00
|
|
|
|
2012-10-19 21:20:45 +04:00
|
|
|
if (twctx->instruction_filter > 0 &&
|
|
|
|
!(twctx->instruction_filter & cur->instruction) ) {
|
2012-03-19 19:05:23 +04:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2017-08-17 15:39:23 +03:00
|
|
|
visitor = (csync_treewalk_visit_func*)(twctx->user_visitor);
|
2012-10-19 21:20:45 +04:00
|
|
|
if (visitor != NULL) {
|
2017-08-17 15:39:23 +03:00
|
|
|
rc = (*visitor)(cur, other, twctx->userdata);
|
2014-03-24 14:35:19 +04:00
|
|
|
|
2012-12-15 19:34:31 +04:00
|
|
|
return rc;
|
2012-03-19 19:05:23 +04:00
|
|
|
}
|
2013-03-12 20:01:39 +04:00
|
|
|
ctx->status_code = CSYNC_STATUS_PARAM_ERROR;
|
2012-03-19 19:05:23 +04:00
|
|
|
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.
|
|
|
|
*/
|
2017-08-23 20:16:12 +03:00
|
|
|
static int _csync_walk_tree(CSYNC *ctx, csync_s::FileMap *tree, csync_treewalk_visit_func *visitor, int filter)
|
2012-03-19 19:05:23 +04:00
|
|
|
{
|
|
|
|
_csync_treewalk_context tw_ctx;
|
2017-08-23 20:16:12 +03:00
|
|
|
int rc = 0;
|
2012-04-17 14:31:27 +04:00
|
|
|
|
2012-10-27 17:10:16 +04:00
|
|
|
tw_ctx.userdata = ctx->callbacks.userdata;
|
2012-03-19 19:05:23 +04:00
|
|
|
tw_ctx.user_visitor = visitor;
|
|
|
|
tw_ctx.instruction_filter = filter;
|
|
|
|
|
2012-10-27 17:10:16 +04:00
|
|
|
ctx->callbacks.userdata = &tw_ctx;
|
2012-03-19 19:05:23 +04:00
|
|
|
|
2017-08-23 20:16:12 +03:00
|
|
|
for (auto &pair : *tree) {
|
|
|
|
if (_csync_treewalk_visitor(pair.second.get(), ctx) < 0) {
|
|
|
|
rc = -1;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-12-17 20:22:24 +04:00
|
|
|
if( rc < 0 ) {
|
2013-08-18 18:21:18 +04:00
|
|
|
if( ctx->status_code == CSYNC_STATUS_OK )
|
2013-08-18 19:26:45 +04:00
|
|
|
ctx->status_code = csync_errno_to_status(errno, CSYNC_STATUS_TREE_ERROR);
|
2012-12-17 20:22:24 +04:00
|
|
|
}
|
2012-10-27 17:10:16 +04:00
|
|
|
ctx->callbacks.userdata = tw_ctx.userdata;
|
2012-03-19 19:05:23 +04:00
|
|
|
|
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* wrapper function for treewalk on the remote tree
|
|
|
|
*/
|
|
|
|
int csync_walk_remote_tree(CSYNC *ctx, csync_treewalk_visit_func *visitor, int filter)
|
|
|
|
{
|
2017-08-23 20:16:12 +03:00
|
|
|
ctx->status_code = CSYNC_STATUS_OK;
|
|
|
|
ctx->current = REMOTE_REPLICA;
|
|
|
|
return _csync_walk_tree(ctx, &ctx->remote.files, visitor, filter);
|
2012-03-19 19:05:23 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* wrapper function for treewalk on the local tree
|
|
|
|
*/
|
|
|
|
int csync_walk_local_tree(CSYNC *ctx, csync_treewalk_visit_func *visitor, int filter)
|
|
|
|
{
|
2017-08-23 20:16:12 +03:00
|
|
|
ctx->status_code = CSYNC_STATUS_OK;
|
|
|
|
ctx->current = LOCAL_REPLICA;
|
|
|
|
return _csync_walk_tree(ctx, &ctx->local.files, visitor, filter);
|
2008-04-28 15:42:10 +04:00
|
|
|
}
|
|
|
|
|
2017-09-04 20:06:13 +03:00
|
|
|
int csync_s::reinitialize() {
|
2013-04-20 12:51:27 +04:00
|
|
|
int rc = 0;
|
|
|
|
|
2017-09-04 20:06:13 +03:00
|
|
|
status_code = CSYNC_STATUS_OK;
|
2013-05-08 17:28:26 +04:00
|
|
|
|
2017-09-04 20:06:13 +03:00
|
|
|
if (statedb.db != NULL
|
|
|
|
&& csync_statedb_close(this) < 0) {
|
2017-03-31 17:13:01 +03:00
|
|
|
CSYNC_LOG(CSYNC_LOG_PRIORITY_WARN, "ERR: closing of statedb failed.");
|
2013-10-31 14:14:21 +04:00
|
|
|
rc = -1;
|
|
|
|
}
|
2017-09-04 20:06:13 +03:00
|
|
|
statedb.db = NULL;
|
|
|
|
|
|
|
|
remote.read_from_db = 0;
|
|
|
|
read_remote_from_db = true;
|
|
|
|
db_is_empty = false;
|
2013-10-31 13:05:15 +04:00
|
|
|
|
2017-08-23 20:16:12 +03:00
|
|
|
local.files.clear();
|
|
|
|
remote.files.clear();
|
2013-05-06 19:14:17 +04:00
|
|
|
|
2017-09-04 20:06:13 +03:00
|
|
|
status = CSYNC_STATUS_INIT;
|
|
|
|
SAFE_FREE(error_string);
|
2013-04-22 17:39:43 +04:00
|
|
|
|
2013-04-24 16:35:19 +04:00
|
|
|
rc = 0;
|
2013-04-20 12:51:27 +04:00
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
|
2017-09-04 20:06:13 +03:00
|
|
|
csync_s::~csync_s() {
|
|
|
|
if (statedb.db != NULL
|
|
|
|
&& csync_statedb_close(this) < 0) {
|
2017-03-31 17:13:01 +03:00
|
|
|
CSYNC_LOG(CSYNC_LOG_PRIORITY_WARN, "ERR: closing of statedb failed.");
|
2013-10-31 14:14:21 +04:00
|
|
|
}
|
2017-09-04 20:06:13 +03:00
|
|
|
statedb.db = NULL;
|
2013-10-31 14:14:21 +04:00
|
|
|
|
2017-09-04 20:06:13 +03:00
|
|
|
SAFE_FREE(statedb.file);
|
|
|
|
SAFE_FREE(local.uri);
|
|
|
|
SAFE_FREE(error_string);
|
2008-02-27 20:56:47 +03:00
|
|
|
}
|
|
|
|
|
2009-03-26 13:09:46 +03:00
|
|
|
void *csync_get_userdata(CSYNC *ctx) {
|
|
|
|
if (ctx == NULL) {
|
|
|
|
return NULL;
|
|
|
|
}
|
2012-10-27 17:10:16 +04:00
|
|
|
return ctx->callbacks.userdata;
|
2009-03-26 13:09:46 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
int csync_set_userdata(CSYNC *ctx, void *userdata) {
|
|
|
|
if (ctx == NULL) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2012-10-27 17:10:16 +04:00
|
|
|
ctx->callbacks.userdata = userdata;
|
2009-03-26 13:09:46 +03:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2008-06-24 15:36:27 +04:00
|
|
|
csync_auth_callback csync_get_auth_callback(CSYNC *ctx) {
|
2008-05-20 18:33:03 +04:00
|
|
|
if (ctx == NULL) {
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2012-10-27 17:10:16 +04:00
|
|
|
return ctx->callbacks.auth_function;
|
2012-10-27 17:15:59 +04:00
|
|
|
}
|
|
|
|
|
2008-05-20 18:33:03 +04:00
|
|
|
int csync_set_status(CSYNC *ctx, int status) {
|
|
|
|
if (ctx == NULL || status < 0) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2008-05-05 13:09:16 +04:00
|
|
|
ctx->status = status;
|
2008-05-20 18:33:03 +04:00
|
|
|
|
|
|
|
return 0;
|
2008-05-05 13:09:16 +04:00
|
|
|
}
|
|
|
|
|
2013-11-25 01:17:36 +04:00
|
|
|
CSYNC_STATUS csync_get_status(CSYNC *ctx) {
|
2008-06-24 13:13:56 +04:00
|
|
|
if (ctx == NULL) {
|
2017-08-14 17:19:52 +03:00
|
|
|
return CSYNC_STATUS_ERROR;
|
2008-06-24 13:13:56 +04:00
|
|
|
}
|
|
|
|
|
2013-11-25 01:17:36 +04:00
|
|
|
return ctx->status_code;
|
2012-02-27 15:18:02 +04:00
|
|
|
}
|
|
|
|
|
2013-03-13 17:46:08 +04:00
|
|
|
const char *csync_get_status_string(CSYNC *ctx)
|
2012-12-04 16:42:16 +04:00
|
|
|
{
|
2013-03-13 17:46:08 +04:00
|
|
|
return csync_vio_get_status_string(ctx);
|
2012-12-04 16:42:16 +04:00
|
|
|
}
|
|
|
|
|
2013-05-08 19:33:50 +04:00
|
|
|
void csync_request_abort(CSYNC *ctx)
|
2013-05-08 17:28:26 +04:00
|
|
|
{
|
2013-05-16 19:37:30 +04:00
|
|
|
if (ctx != NULL) {
|
2013-05-08 17:28:26 +04:00
|
|
|
ctx->abort = true;
|
2013-05-16 19:37:30 +04:00
|
|
|
}
|
2013-05-08 17:28:26 +04:00
|
|
|
}
|
|
|
|
|
2013-05-08 19:33:50 +04:00
|
|
|
void csync_resume(CSYNC *ctx)
|
|
|
|
{
|
2013-05-16 19:37:30 +04:00
|
|
|
if (ctx != NULL) {
|
2013-05-08 19:33:50 +04:00
|
|
|
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);
|
|
|
|
}
|
2013-05-08 19:33:50 +04:00
|
|
|
}
|