nextcloud-desktop/src/csync/csync.cpp

Ignoring revisions in .git-blame-ignore-revs. Click here to bypass and see the normal blame view.

545 lines
13 KiB
C++
Raw Normal View History

2008-02-27 20:56:47 +03:00
/*
* libcsync -- a library to sync a directory with another
*
* 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
*
* 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
*
* This library 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
* Lesser General Public License for more details.
2008-02-27 20:56:47 +03: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
*/
#include "config_csync.h"
2012-03-02 19:47:34 +04:00
#ifndef _GNU_SOURCE
#define _GNU_SOURCE
#endif
2008-04-28 14:40:32 +04:00
#include <assert.h>
2008-02-27 20:56:47 +03:00
#include <errno.h>
#include <stdio.h>
#include <string.h>
#include <time.h>
#include <sys/types.h>
#include <stdbool.h>
2008-02-27 20:56:47 +03:00
#include "c_lib.h"
#include "csync_private.h"
#include "csync_exclude.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 "std/c_private.h"
#include "csync_update.h"
2008-05-15 15:50:34 +04:00
#include "csync_reconcile.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"
#include "c_jhash.h"
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;
}
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 */
len = strlen(localUri);
while(len > 0 && localUri[len - 1] == '/') --len;
2008-02-27 20:56:47 +03:00
local.uri = c_strndup(localUri, len);
2013-04-20 14:06:59 +04:00
c_rbtree_create(&local.tree, _key_cmp, _data_cmp);
c_rbtree_create(&remote.tree, _key_cmp, _data_cmp);
statedb.file = c_strdup(db_file);
2008-02-27 20:56:47 +03:00
}
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
if (ctx == NULL) {
errno = EBADF;
return -1;
}
2013-08-18 19:26:45 +04:00
ctx->status_code = CSYNC_STATUS_OK;
/* Path of database file is set in csync_init */
if (csync_statedb_load(ctx, ctx->statedb.file, &ctx->statedb.db) < 0) {
rc = -1;
return rc;
}
ctx->status_code = CSYNC_STATUS_OK;
csync_memstat_check();
if (!ctx->excludes) {
CSYNC_LOG(CSYNC_LOG_PRIORITY_INFO, "No exclude file loaded or defined!");
}
/* 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;
rc = csync_ftw(ctx, ctx->local.uri, csync_walker, MAX_DEPTH);
if (rc < 0) {
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);
}
goto out;
}
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,
"Update detection for local replica took %.2f seconds walking %zu files.",
c_secdiff(finish, start), c_rbtree_size(ctx->local.tree));
csync_memstat_check();
/* update detection for remote replica */
csync_gettime(&start);
ctx->current = REMOTE_REPLICA;
rc = csync_ftw(ctx, "", csync_walker, MAX_DEPTH);
if (rc < 0) {
if(ctx->status_code == CSYNC_STATUS_OK) {
ctx->status_code = csync_errno_to_status(errno, CSYNC_STATUS_UPDATE_ERROR);
}
goto out;
}
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();
ctx->status |= CSYNC_STATUS_UPDATE;
rc = 0;
out:
csync_statedb_close(ctx);
return rc;
}
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->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
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.",
c_secdiff(finish, start), c_rbtree_size(ctx->local.tree));
2008-05-15 15:50:34 +04:00
if (rc < 0) {
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 );
}
goto out;
2008-05-15 15:50:34 +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
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.",
c_secdiff(finish, start), c_rbtree_size(ctx->remote.tree));
2008-05-15 15:50:34 +04:00
if (rc < 0) {
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 );
}
goto out;
2008-05-15 15:50:34 +04:00
}
ctx->status |= CSYNC_STATUS_RECONCILE;
2008-05-15 15:50:34 +04:00
rc = 0;
out:
csync_statedb_close(ctx);
2008-05-15 15:50:34 +04:00
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;
csync_treewalk_visit_func *visitor = NULL;
_csync_treewalk_context *twctx = NULL;
c_rbtree_t *other_tree = NULL;
c_rbnode_t *other_node = NULL;
cur = (csync_file_stat_t *) obj;
ctx = (CSYNC *) data;
if (ctx == NULL) {
2013-04-05 17:05:44 +04:00
return -1;
}
/* we need the opposite tree! */
switch (ctx->current) {
case LOCAL_REPLICA:
other_tree = ctx->remote.tree;
break;
case REMOTE_REPLICA:
other_tree = ctx->local.tree;
break;
default:
break;
}
other_node = c_rbtree_find(other_tree, &cur->phash);
if (!other_node) {
/* Check the renamed path as well. */
int len;
uint64_t h = 0;
char *renamed_path = csync_rename_adjust_path(ctx, cur->path);
if (!c_streq(renamed_path, cur->path)) {
len = strlen( renamed_path );
h = c_jhash64((uint8_t *) renamed_path, len, 0);
other_node = c_rbtree_find(other_tree, &h);
}
SAFE_FREE(renamed_path);
}
if (!other_node) {
/* Check the source path as well. */
int len;
uint64_t h = 0;
char *renamed_path = csync_rename_adjust_path_source(ctx, cur->path);
if (!c_streq(renamed_path, cur->path)) {
len = strlen( renamed_path );
h = c_jhash64((uint8_t *) renamed_path, len, 0);
other_node = c_rbtree_find(other_tree, &h);
}
SAFE_FREE(renamed_path);
}
csync_file_stat_t *other = other_node ? (csync_file_stat_t*)other_node->data : NULL;
if (obj == NULL || data == NULL) {
2013-08-18 19:26:45 +04:00
ctx->status_code = CSYNC_STATUS_PARAM_ERROR;
return -1;
}
2013-08-18 19:26:45 +04:00
ctx->status_code = CSYNC_STATUS_OK;
twctx = (_csync_treewalk_context*) ctx->callbacks.userdata;
if (twctx == NULL) {
ctx->status_code = CSYNC_STATUS_PARAM_ERROR;
return -1;
}
if (twctx->instruction_filter > 0 &&
!(twctx->instruction_filter & cur->instruction) ) {
return 0;
}
visitor = (csync_treewalk_visit_func*)(twctx->user_visitor);
if (visitor != NULL) {
rc = (*visitor)(cur, other, twctx->userdata);
return rc;
}
ctx->status_code = CSYNC_STATUS_PARAM_ERROR;
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 rc;
}
if (visitor == NULL || tree == NULL) {
ctx->status_code = CSYNC_STATUS_PARAM_ERROR;
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->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);
}
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 != NULL) {
ctx->status_code = CSYNC_STATUS_OK;
ctx->current = REMOTE_REPLICA;
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 != NULL) {
ctx->status_code = CSYNC_STATUS_OK;
ctx->current = LOCAL_REPLICA;
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;
delete freedata;
}
int csync_s::reinitialize() {
int rc = 0;
status_code = CSYNC_STATUS_OK;
if (statedb.db != NULL
&& csync_statedb_close(this) < 0) {
CSYNC_LOG(CSYNC_LOG_PRIORITY_WARN, "ERR: closing of statedb failed.");
rc = -1;
}
statedb.db = NULL;
/* destroy the rbtrees */
if (c_rbtree_size(local.tree) > 0) {
c_rbtree_destroy(local.tree, _tree_destructor);
}
if (c_rbtree_size(remote.tree) > 0) {
c_rbtree_destroy(remote.tree, _tree_destructor);
}
/* free memory */
c_rbtree_free(local.tree);
c_rbtree_free(remote.tree);
remote.read_from_db = 0;
read_remote_from_db = true;
db_is_empty = false;
/* Create new trees */
c_rbtree_create(&local.tree, _key_cmp, _data_cmp);
c_rbtree_create(&remote.tree, _key_cmp, _data_cmp);
status = CSYNC_STATUS_INIT;
SAFE_FREE(error_string);
rc = 0;
return rc;
}
csync_s::~csync_s() {
if (statedb.db != NULL
&& csync_statedb_close(this) < 0) {
CSYNC_LOG(CSYNC_LOG_PRIORITY_WARN, "ERR: closing of statedb failed.");
}
statedb.db = NULL;
/* destroy the rbtrees */
if (c_rbtree_size(local.tree) > 0) {
c_rbtree_destroy(local.tree, _tree_destructor);
}
2013-01-04 23:45:10 +04:00
if (c_rbtree_size(remote.tree) > 0) {
c_rbtree_destroy(remote.tree, _tree_destructor);
}
2008-02-27 20:56:47 +03:00
/* free memory */
c_rbtree_free(local.tree);
c_rbtree_free(remote.tree);
2008-02-27 20:56:47 +03:00
SAFE_FREE(statedb.file);
SAFE_FREE(local.uri);
SAFE_FREE(error_string);
2008-02-27 20:56:47 +03:00
}
void *csync_get_userdata(CSYNC *ctx) {
if (ctx == NULL) {
return NULL;
}
return ctx->callbacks.userdata;
}
int csync_set_userdata(CSYNC *ctx, void *userdata) {
if (ctx == NULL) {
return -1;
}
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;
}
return ctx->callbacks.auth_function;
2012-10-27 17:15:59 +04:00
}
int csync_set_status(CSYNC *ctx, int status) {
if (ctx == NULL || status < 0) {
return -1;
}
ctx->status = status;
return 0;
}
CSYNC_STATUS csync_get_status(CSYNC *ctx) {
2008-06-24 13:13:56 +04:00
if (ctx == NULL) {
return CSYNC_STATUS_ERROR;
2008-06-24 13:13:56 +04:00
}
return ctx->status_code;
}
const char *csync_get_status_string(CSYNC *ctx)
{
return csync_vio_get_status_string(ctx);
}
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);
}
}