/* * 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 "csync_private.h" #include "csync_propagate.h" #include "csync_statedb.h" #include "vio/csync_vio_local.h" #include "vio/csync_vio.h" #include "c_jhash.h" #define CSYNC_LOG_CATEGORY_NAME "csync.propagator" #include "csync_log.h" #include "csync_util.h" #include "csync_misc.h" #include "csync_rename.h" static int _csync_build_remote_uri(CSYNC *ctx, char **dst, const char *path) { char *tmp = csync_rename_adjust_path(ctx, path); int ret = asprintf(dst, "%s/%s", ctx->remote.uri, tmp); SAFE_FREE(tmp); return ret; } static int _csync_cleanup_cmp(const void *a, const void *b) { csync_file_stat_t *st_a, *st_b; st_a = *((csync_file_stat_t **) a); st_b = *((csync_file_stat_t **) b); return strcmp(st_a->path, st_b->path); } static void _csync_file_stat_set_error(csync_file_stat_t *st, const char *error) { st->instruction = CSYNC_INSTRUCTION_ERROR; if (st->error_string || !error) return; // do not override first error. st->error_string = c_strdup(error); } /* Recursively mark the parent flder as an error */ static void _csync_report_parent_error(CSYNC *ctx, csync_file_stat_t *st) { const char *dir = NULL; uint64_t h; c_rbnode_t* node; dir = c_dirname(st->path); if (!dir) return; h = c_jhash64((uint8_t *) dir, strlen(dir), 0); node = c_rbtree_find(ctx->local.tree, &h); if (!node) { /* Not in the local tree, mark the remote tree as an error then */ node = c_rbtree_find(ctx->remote.tree, &h); } if (node) { st = node->data; CSYNC_LOG(CSYNC_LOG_PRIORITY_TRACE, "Mark parent directoy `%s` as an error", dir); _csync_file_stat_set_error(st, "Error within the directory"); _csync_report_parent_error(ctx, st); } SAFE_FREE(dir); } /* Record the error in the ctx->progress pi may be a previous csync_progressinfo_t from the database. If pi is NULL, a new one is created, else it is re-used */ static void _csync_record_error(CSYNC *ctx, csync_file_stat_t *st, csync_progressinfo_t *pi) { _csync_file_stat_set_error(st, csync_get_error_string(ctx)); _csync_report_parent_error(ctx, st); if (pi) { pi->error++; SAFE_FREE(pi->error_string); } else { pi = c_malloc(sizeof(csync_progressinfo_t)); pi->chunk = 0; pi->transferId = 0; pi->tmpfile = NULL; pi->md5 = st->md5 ? c_strdup(st->md5) : NULL; pi->modtime = st->modtime; pi->phash = st->phash; pi->error = 1; } pi->error_string = st->error_string ? c_strdup(st->error_string) : NULL; pi->next = ctx->progress_info; ctx->progress_info = pi; } static bool _push_to_tmp_first(CSYNC *ctx) { if( !ctx ) return true; if( ctx->current == REMOTE_REPLICA ) return true; /* Always push to tmp for destination local file system */ /* If destination is the remote replica check if the switch is set. */ if( !ctx->module.capabilities.atomar_copy_support ) return true; return false; } static void _notify_progress(CSYNC *ctx, const char *file, int64_t filesize, enum csync_notify_type_e kind) { if (ctx == NULL) { return; } if (ctx->callbacks.progress_cb) { CSYNC_PROGRESS progress; progress.kind = kind; progress.path = file; progress.curr_bytes = 0; progress.file_size = filesize; progress.overall_transmission_size = ctx->overall_progress.byte_sum; progress.current_overall_bytes = ctx->overall_progress.byte_current; progress.overall_file_count = ctx->overall_progress.file_count; progress.current_file_no = ctx->overall_progress.current_file_no; ctx->callbacks.progress_cb(&progress, ctx->callbacks.userdata); } } static bool _use_fd_based_push(CSYNC *ctx) { if(!ctx) return false; if( ctx->module.capabilities.use_send_file_to_propagate ) return true; return false; } static const char*_get_md5( CSYNC *ctx, const char *path ) { const char *md5 = NULL; char *buf = NULL; /* Always use the remote uri path, local does not have Ids. */ if (asprintf(&buf, "%s/%s", ctx->remote.uri, path) < 0) { return 0; } md5 = csync_vio_file_id(ctx, buf); CSYNC_LOG(CSYNC_LOG_PRIORITY_DEBUG, "MD5 for %s: %s", buf, md5 ? md5 : ""); SAFE_FREE(buf); return md5; } static int _csync_push_file(CSYNC *ctx, csync_file_stat_t *st) { enum csync_replica_e srep = -1; enum csync_replica_e drep = -1; enum csync_replica_e rep_bak = -1; char *suri = NULL; char *duri = NULL; char *turi = NULL; char *tdir = NULL; char *auri = NULL; const char *tmd5 = NULL; char *prev_tdir = NULL; csync_vio_handle_t *sfp = NULL; csync_vio_handle_t *dfp = NULL; csync_vio_file_stat_t *tstat = NULL; csync_vio_file_stat_t *vst = NULL; char errbuf[256] = {0}; char buf[MAX_XFER_BUF_SIZE] = {0}; ssize_t bread = 0; ssize_t bwritten = 0; struct timeval times[2]; int rc = -1; int count = 0; int flags = 0; bool do_pre_copy_stat = false; /* do an additional stat before actually copying */ csync_hbf_info_t hbf_info = { 0, 0 }; csync_progressinfo_t *progress_info = NULL; enum csync_notify_type_e notify_start_kind = CSYNC_NOTIFY_START_UPLOAD; enum csync_notify_type_e notify_end_kind = CSYNC_NOTIFY_FINISHED_UPLOAD; /* Check if there is progress info stored in the database for this file */ progress_info = csync_statedb_get_progressinfo(ctx, st->phash, st->modtime, st->md5); rep_bak = ctx->replica; #ifdef BLACKLIST_ON_ERROR if (progress_info && progress_info->error > 3) { CSYNC_LOG(CSYNC_LOG_PRIORITY_ERROR, "'%s' was blacklisted after %d errors: %s", st->path, progress_info->error, progress_info->error_string); rc = 1; goto out; } #endif if (progress_info) { CSYNC_LOG(CSYNC_LOG_PRIORITY_TRACE, "continuation: %d %d", progress_info->chunk, progress_info->transferId ); hbf_info.start_id = progress_info->chunk; hbf_info.transfer_id = progress_info->transferId; } auri = csync_rename_adjust_path(ctx, st->path); switch (ctx->current) { case LOCAL_REPLICA: srep = ctx->local.type; drep = ctx->remote.type; if (asprintf(&suri, "%s/%s", ctx->local.uri, auri) < 0) { rc = -1; goto out; } if (_csync_build_remote_uri(ctx, &duri, st->path) < 0) { rc = -1; goto out; } do_pre_copy_stat = true; break; case REMOTE_REPLICA: srep = ctx->remote.type; drep = ctx->local.type; notify_start_kind = CSYNC_NOTIFY_START_DOWNLOAD; notify_end_kind = CSYNC_NOTIFY_FINISHED_DOWNLOAD; if (_csync_build_remote_uri(ctx, &suri, st->path) < 0) { rc = -1; goto out; } if (asprintf(&duri, "%s/%s", ctx->local.uri, auri) < 0) { rc = -1; goto out; } break; default: break; } /* Increment by one as its started now. */ ctx->overall_progress.current_file_no++; _notify_progress(ctx, duri, 0, notify_start_kind); /* Check if the file is still untouched since the update run. */ if (do_pre_copy_stat) { vst = csync_vio_file_stat_new(); if (csync_vio_stat(ctx, suri, vst) < 0) { /* Pre copy stat failed */ rc = 1; goto out; } else { /* Pre copy stat succeeded */ if (st->modtime != vst->mtime || st->size != vst->size) { /* The size or modtime has changed. Skip this file copy for now. */ rc = 1; /* soft problem */ CSYNC_LOG(CSYNC_LOG_PRIORITY_DEBUG, "Source file has changed since update run, SKIP it for now."); goto out; } csync_vio_file_stat_destroy(vst); } } /* Open the source file */ ctx->replica = srep; flags = O_RDONLY|O_NOFOLLOW; #ifdef O_NOATIME /* O_NOATIME can only be set by the owner of the file or the superuser */ if (st->uid == ctx->pwd.uid || ctx->pwd.euid == 0) { flags |= O_NOATIME; } #endif sfp = csync_vio_open(ctx, suri, flags, 0); if (sfp == NULL) { if (errno == ENOMEM) { rc = -1; } else { rc = 1; } C_STRERROR(errno, errbuf, sizeof(errbuf)); CSYNC_LOG(CSYNC_LOG_PRIORITY_ERROR, "file: %s, command: open(O_RDONLY), error: %s", suri, errbuf ); goto out; } if (_push_to_tmp_first(ctx)) { if (progress_info && progress_info->tmpfile && progress_info->tmpfile[0] && _push_to_tmp_first(ctx)) { turi = c_strdup(progress_info->tmpfile); /* Try to see if we can resume. */ ctx->replica = drep; dfp = csync_vio_open(ctx, turi, O_WRONLY|O_APPEND|O_NOCTTY, 0); if (dfp) { goto start_fd_based; } } /* create the temporary file name */ turi = c_tmpname(duri); if (!turi) { rc = -1; goto out; } } else { /* write to the target file directly as the HTTP server does it atomically */ if (asprintf(&turi, "%s", duri) < 0) { rc = -1; goto out; } CSYNC_LOG(CSYNC_LOG_PRIORITY_TRACE, "Remote repository atomar push enabled for %s (%d).", turi, ctx->current); } /* Create the destination file */ ctx->replica = drep; while ((dfp = csync_vio_open(ctx, turi, O_CREAT|O_EXCL|O_WRONLY|O_NOCTTY, C_FILE_MODE)) == NULL) { CSYNC_LOG(CSYNC_LOG_PRIORITY_TRACE, "file: %s, command: open(O_CREAT), error: %d", duri, errno); switch (errno) { case EEXIST: if (count++ > 10) { CSYNC_LOG(CSYNC_LOG_PRIORITY_ERROR, "file: %s, command: open(O_CREAT), error: max count exceeded", duri); rc = 1; goto out; } if(_push_to_tmp_first(ctx)) { SAFE_FREE(turi); turi = c_tmpname(duri); if (!turi) { rc = -1; goto out; } } break; case ENOENT: /* get the directory name */ SAFE_FREE(tdir); tdir = c_dirname(turi); if (tdir == NULL) { rc = -1; goto out; } if( prev_tdir && c_streq(tdir, prev_tdir) ) { /* we're looping */ CSYNC_LOG(CSYNC_LOG_PRIORITY_WARN, "dir: %s, loop in mkdir detected!", tdir); rc = 1; goto out; } SAFE_FREE(prev_tdir); prev_tdir = c_strdup(tdir); if (csync_vio_mkdirs(ctx, tdir, C_DIR_MODE) < 0) { C_STRERROR(errno, errbuf, sizeof(errbuf)); CSYNC_LOG(CSYNC_LOG_PRIORITY_WARN, "dir: %s, command: mkdirs, error: %s", tdir, errbuf); } break; case ENOMEM: rc = -1; C_STRERROR(errno, errbuf, sizeof(errbuf)); CSYNC_LOG(CSYNC_LOG_PRIORITY_ERROR, "file: %s, command: open(O_CREAT), error: %s", turi, errbuf); goto out; break; default: C_STRERROR(errno, errbuf, sizeof(errbuf)); CSYNC_LOG(CSYNC_LOG_PRIORITY_ERROR, "file: %s, command: open(O_CREAT), error: %s", turi, errbuf); rc = 1; goto out; break; } } /* copy file */ if( _use_fd_based_push(ctx) ) { start_fd_based: if (ctx->current == REMOTE_REPLICA) { csync_win32_set_file_hidden(turi, true); } if (!_push_to_tmp_first(ctx)) { csync_vio_set_property(ctx, "hbf_info", &hbf_info); } rc = csync_vio_sendfile( ctx, sfp, dfp ); if (ctx->current == REMOTE_REPLICA) { csync_win32_set_file_hidden(turi, false); } if( rc != 0 ) { if (rc == -1) { /* Severe error */ switch(errno) { case EINVAL: ctx->error_code = CSYNC_ERR_PARAM; break; case ERRNO_USER_ABORT: ctx->error_code = CSYNC_ERR_ABORTED; break; case ERRNO_GENERAL_ERROR: default: ctx->error_code = CSYNC_ERR_PROPAGATE; break; } /* fetch the error string from module. */ ctx->error_string = csync_vio_get_error_string(ctx); } C_STRERROR(errno, errbuf, sizeof(errbuf)); CSYNC_LOG(CSYNC_LOG_PRIORITY_ERROR, "file: %s, command: sendfile, error: %s from errno %d", suri, c_streq(errbuf, "") ? csync_vio_get_error_string(ctx): errbuf, errno); if (_push_to_tmp_first(ctx)) { csync_vio_file_stat_t* sb = csync_vio_file_stat_new(); if (csync_vio_stat(ctx, turi, sb) == 0 && sb->size > 0 && errno != EIO) { /* EIO is mapped to error from owncloud like 500 for which we don't want to resume */ CSYNC_LOG(CSYNC_LOG_PRIORITY_TRACE, "keeping tmp file: %s", turi); if (!progress_info) { progress_info = c_malloc(sizeof(csync_progressinfo_t)); progress_info->error = 0; progress_info->transferId = 0; progress_info->md5 = st->md5 ? c_strdup(st->md5) : NULL; progress_info->modtime = st->modtime; progress_info->phash = st->phash; } else { SAFE_FREE(progress_info->tmpfile); } progress_info->chunk = 0; progress_info->tmpfile = turi; progress_info->error <<= 1; turi = NULL; } csync_vio_file_stat_destroy(sb); } else { CSYNC_LOG(CSYNC_LOG_PRIORITY_TRACE, "remember chunk: %d (transfer id %d )", hbf_info.start_id, hbf_info.transfer_id); if (!progress_info) { progress_info = c_malloc(sizeof(csync_progressinfo_t)); progress_info->error = 0; progress_info->md5 = st->md5 ? c_strdup(st->md5) : NULL; progress_info->modtime = st->modtime; progress_info->phash = st->phash; progress_info->tmpfile = NULL; } else { SAFE_FREE(progress_info->tmpfile); } progress_info->transferId = hbf_info.transfer_id; progress_info->chunk = hbf_info.start_id; csync_vio_set_property(ctx, "hbf_info", 0); } if( errno == ERRNO_USER_ABORT ) { ctx->error_code = CSYNC_ERR_ABORTED; CSYNC_LOG(CSYNC_LOG_PRIORITY_TRACE, "Csync file transmission was ABORTED by user!"); } goto out; } } else { for (;;) { ctx->replica = srep; bread = csync_vio_read(ctx, sfp, buf, MAX_XFER_BUF_SIZE); if (bread < 0) { /* read error */ C_STRERROR(errno, errbuf, sizeof(errbuf)); CSYNC_LOG(CSYNC_LOG_PRIORITY_ERROR, "file: %s, command: read, error: %s", suri, errbuf); rc = 1; goto out; } else if (bread == 0) { /* done */ break; } ctx->replica = drep; bwritten = csync_vio_write(ctx, dfp, buf, bread); if (bwritten < 0 || bread != bwritten) { C_STRERROR(errno, errbuf, sizeof(errbuf)); CSYNC_LOG(CSYNC_LOG_PRIORITY_ERROR, "file: %s, command: write, error: bread = %zu, bwritten = %zu - %s", duri, bread, bwritten, errbuf); rc = 1; goto out; } } } ctx->replica = srep; if (csync_vio_close(ctx, sfp) < 0) { C_STRERROR(errno, errbuf, sizeof(errbuf)); CSYNC_LOG(CSYNC_LOG_PRIORITY_ERROR, "file: %s, command: close, error: %s", suri, errbuf); } sfp = NULL; ctx->replica = drep; if (csync_vio_close(ctx, dfp) < 0) { dfp = NULL; switch (errno) { /* stop if no space left or quota exceeded */ case ENOSPC: case EDQUOT: C_STRERROR(errno, errbuf, sizeof(errbuf)); CSYNC_LOG(CSYNC_LOG_PRIORITY_ERROR, "file: %s, command: close, error: %s", turi, errbuf); rc = -1; goto out; break; default: C_STRERROR(errno, errbuf, sizeof(errbuf)); CSYNC_LOG(CSYNC_LOG_PRIORITY_ERROR, "file: %s, command: close, error: %s", turi, errbuf); break; } } dfp = NULL; if( ctx->module.capabilities.do_post_copy_stat ) { /* * Check filesize * In case the transport is secure and/or the stat is expensive, this check * could be skipped through module capabilities definitions. */ ctx->replica = drep; tstat = csync_vio_file_stat_new(); if (tstat == NULL) { C_STRERROR(errno, errbuf, sizeof(errbuf)); CSYNC_LOG(CSYNC_LOG_PRIORITY_ERROR, "file: %s, command: stat, error: %s", turi, errbuf); rc = -1; goto out; } if (csync_vio_stat(ctx, turi, tstat) < 0) { switch (errno) { case ENOMEM: rc = -1; break; default: rc = 1; break; } C_STRERROR(errno, errbuf, sizeof(errbuf)); CSYNC_LOG(CSYNC_LOG_PRIORITY_ERROR, "file: %s, command: stat, error: %s", turi, errbuf); goto out; } if (st->size != tstat->size) { CSYNC_LOG(CSYNC_LOG_PRIORITY_ERROR, "file: %s, error: incorrect filesize (size: %" PRId64 " should be %" PRId64 ")", turi, tstat->size, st->size); rc = 1; goto out; } if( st->md5 ) { CSYNC_LOG(CSYNC_LOG_PRIORITY_DEBUG, "UUUU MD5 sum: %s", st->md5); } else { if( tstat->md5 ) { CSYNC_LOG(CSYNC_LOG_PRIORITY_DEBUG, "Target MD5 sum is %s", tstat->md5 ); if(st->md5) SAFE_FREE(st->md5); st->md5 = c_strdup(tstat->md5 ); } else { CSYNC_LOG(CSYNC_LOG_PRIORITY_DEBUG, "MD5 sum is empty"); } } } if (_push_to_tmp_first(ctx)) { /* override original file */ ctx->replica = drep; if (csync_vio_rename(ctx, turi, duri) < 0) { switch (errno) { case ENOMEM: rc = -1; break; default: rc = 1; break; } C_STRERROR(errno, errbuf, sizeof(errbuf)); CSYNC_LOG(CSYNC_LOG_PRIORITY_ERROR, "file: %s, command: rename, error: %s", duri, errbuf); goto out; } } /* set mode only if it is not the default mode */ if ((st->mode & 07777) != C_FILE_MODE) { if (csync_vio_chmod(ctx, duri, st->mode) < 0) { switch (errno) { case ENOMEM: rc = -1; break; default: rc = 1; break; } C_STRERROR(errno, errbuf, sizeof(errbuf)); CSYNC_LOG(CSYNC_LOG_PRIORITY_ERROR, "file: %s, command: chmod, error: %s", duri, errbuf); goto out; } } /* set owner and group if possible */ if (ctx->pwd.euid == 0) { csync_vio_chown(ctx, duri, st->uid, st->gid); } /* sync time */ times[0].tv_sec = times[1].tv_sec = st->modtime; times[0].tv_usec = times[1].tv_usec = 0; ctx->replica = drep; csync_vio_utimes(ctx, duri, times); /* For remote repos, after the utimes call, the ID has changed again */ /* do a stat on the target again to get a valid md5 */ tmd5 = _get_md5(ctx, auri); CSYNC_LOG(CSYNC_LOG_PRIORITY_DEBUG, "FINAL MD5: %s", tmd5 ? tmd5 : ""); if(tmd5) { SAFE_FREE(st->md5); st->md5 = tmd5; } /* set instruction for the statedb merger */ st->instruction = CSYNC_INSTRUCTION_UPDATED; /* Notify the progress */ ctx->overall_progress.byte_current += st->size; _notify_progress(ctx, duri, st->size, notify_end_kind); CSYNC_LOG(CSYNC_LOG_PRIORITY_DEBUG, "PUSHED file: %s", duri); rc = 0; out: ctx->replica = srep; csync_vio_close(ctx, sfp); ctx->replica = drep; csync_vio_close(ctx, dfp); csync_vio_file_stat_destroy(tstat); /* set instruction for the statedb merger */ if (rc != 0) { if (_push_to_tmp_first(ctx)) { if (turi != NULL) { /* Remove the tmp file in error case. */ csync_vio_unlink(ctx, turi); } } _csync_record_error(ctx, st, progress_info); progress_info = NULL; } csync_statedb_free_progressinfo(progress_info); SAFE_FREE(prev_tdir); SAFE_FREE(suri); SAFE_FREE(duri); SAFE_FREE(turi); SAFE_FREE(tdir); SAFE_FREE(auri); ctx->replica = rep_bak; return rc; } static int _backup_path(CSYNC *ctx, char** duri, const char* uri, const char* path) { int rc=0; C_PATHINFO *info=NULL; struct tm *curtime; time_t sec; char timestring[16]; time(&sec); curtime = localtime(&sec); strftime(timestring, 16, "%Y%m%d-%H%M%S",curtime); info=c_split_path(path); CSYNC_LOG(CSYNC_LOG_PRIORITY_TRACE,"directory: %s",info->directory); CSYNC_LOG(CSYNC_LOG_PRIORITY_TRACE,"filename : %s",info->filename); CSYNC_LOG(CSYNC_LOG_PRIORITY_TRACE,"extension: %s",info->extension); if (asprintf(duri, "%s/%s%s_conflict-%s%s", uri,info->directory ,info->filename,timestring,info->extension) < 0) { rc = -1; } SAFE_FREE(info); return rc; } static int _csync_backup_file(CSYNC *ctx, csync_file_stat_t *st, char **duri) { enum csync_replica_e drep = -1; enum csync_replica_e rep_bak = -1; char *suri = NULL; char errbuf[256] = {0}; int rc = -1; rep_bak = ctx->replica; if(st->instruction==CSYNC_INSTRUCTION_CONFLICT) { CSYNC_LOG(CSYNC_LOG_PRIORITY_TRACE,"CSYNC_INSTRUCTION_CONFLICT"); switch (ctx->current) { case LOCAL_REPLICA: drep = ctx->remote.type; if (asprintf(&suri, "%s/%s", ctx->remote.uri, st->path) < 0) { rc = -1; goto out; } if (_backup_path(ctx, duri, ctx->remote.uri,st->path) < 0) { rc = -1; goto out; } break; case REMOTE_REPLICA: drep = ctx->local.type; if (asprintf(&suri, "%s/%s", ctx->local.uri, st->path) < 0) { rc = -1; goto out; } if ( _backup_path(ctx, duri, ctx->local.uri, st->path) < 0) { rc = -1; goto out; } break; default: break; } } else { CSYNC_LOG(CSYNC_LOG_PRIORITY_TRACE,"instruction not allowed: %i %s",st->instruction,csync_instruction_str(st->instruction)); rc = -1; goto out; } CSYNC_LOG(CSYNC_LOG_PRIORITY_TRACE,"suri: %s",suri); CSYNC_LOG(CSYNC_LOG_PRIORITY_TRACE,"duri: %s",*duri); /* rename the older file to conflict */ ctx->replica = drep; if (csync_vio_rename(ctx, suri, *duri) < 0) { switch (errno) { case ENOMEM: rc = -1; break; default: rc = 1; break; } C_STRERROR(errno, errbuf, sizeof(errbuf)); CSYNC_LOG(CSYNC_LOG_PRIORITY_ERROR, "file: %s, command: rename, error: %s", *duri, errbuf); goto out; } /* set instruction for the statedb merger */ st->instruction = CSYNC_INSTRUCTION_NONE; CSYNC_LOG(CSYNC_LOG_PRIORITY_DEBUG, "BACKUP file: %s", *duri); rc = 0; out: /* set instruction for the statedb merger */ if (rc != 0) { _csync_file_stat_set_error(st, csync_get_error_string(ctx)); } SAFE_FREE(suri); ctx->replica = rep_bak; return rc; } static int _csync_new_file(CSYNC *ctx, csync_file_stat_t *st) { int rc = -1; rc = _csync_push_file(ctx, st); return rc; } static int _csync_rename_file(CSYNC *ctx, csync_file_stat_t *st) { int rc = 0; char errbuf[256] = {0}; struct timeval times[2]; char *suri = NULL; char *duri = NULL; c_rbnode_t *node = NULL; char *tdir = NULL; csync_file_stat_t *other = NULL; csync_progressinfo_t *pi = NULL; /* Find the destination entry in the local tree */ uint64_t h = c_jhash64((uint8_t *) st->destpath, strlen(st->destpath), 0); node = c_rbtree_find(ctx->local.tree, &h); if(node) other = (csync_file_stat_t *) node->data; #ifdef BLACKLIST_ON_ERROR if (other) { pi = csync_statedb_get_progressinfo(ctx, other->phash, other->modtime, other->md5); if (pi && pi->error > 3) { CSYNC_LOG(CSYNC_LOG_PRIORITY_ERROR, "'%s' was blacklisted after %d errors: %s", st->path, pi->error, pi->error_string); if (!st->error_string && pi->error_string) st->error_string = c_strdup(pi->error_string); if (!other->error_string && pi->error_string) other->error_string = c_strdup(pi->error_string); rc = 1; goto out; } } #endif switch (ctx->current) { case REMOTE_REPLICA: if( !(st->path && st->destpath) ) { CSYNC_LOG(CSYNC_LOG_PRIORITY_ERROR, "Rename failed: src or dest path empty"); rc = -1; goto out; } if (_csync_build_remote_uri(ctx, &suri, st->path) < 0) { rc = -1; } if (_csync_build_remote_uri(ctx, &duri, st->destpath) < 0) { rc = -1; } break; case LOCAL_REPLICA: /* No renaming supported by updater */ CSYNC_LOG(CSYNC_LOG_PRIORITY_DEBUG, "RENAME is only supported on local filesystem."); rc = -1; goto out; break; default: break; } if (! c_streq(suri, duri) && rc > -1) { CSYNC_LOG(CSYNC_LOG_PRIORITY_DEBUG, "Renaming %s => %s", suri, duri); while ((rc = csync_vio_rename(ctx, suri, duri)) != 0) { switch (errno) { case ENOENT: /* get the directory name */ if(tdir) { /* we're looping */ CSYNC_LOG(CSYNC_LOG_PRIORITY_WARN, "dir: %s, loop in mkdir detected!", tdir); rc = 1; goto out; } tdir = c_dirname(duri); if (tdir == NULL) { rc = -1; goto out; } if (csync_vio_mkdirs(ctx, tdir, C_DIR_MODE) < 0) { C_STRERROR(errno, errbuf, sizeof(errbuf)); CSYNC_LOG(CSYNC_LOG_PRIORITY_WARN, "dir: %s, command: mkdirs, error: %s", tdir, errbuf); } break; default: C_STRERROR(errno, errbuf, sizeof(errbuf)); CSYNC_LOG(CSYNC_LOG_PRIORITY_ERROR, "dir: %s, command: rename, error: %s", suri, errbuf); goto out; } } /* set owner and group if possible */ if (ctx->pwd.euid == 0) { csync_vio_chown(ctx, duri, st->uid, st->gid); } /* sync time */ times[0].tv_sec = times[1].tv_sec = st->modtime; times[0].tv_usec = times[1].tv_usec = 0; csync_vio_utimes(ctx, duri, times); } if( rc > -1 ) { /* set the mtime which is needed in statedb_get_uniqid */ if( other ) { if (st->type != CSYNC_FTW_TYPE_DIR) { other->md5 = _get_md5(ctx, st->destpath); } else { /* For directories, re-use the old md5 */ other->md5 = c_strdup(st->md5); } } /* set instruction for the statedb merger */ st->instruction = CSYNC_INSTRUCTION_DELETED; } CSYNC_LOG(CSYNC_LOG_PRIORITY_DEBUG, "RENAME file: %s => %s with ID %s", st->path, st->destpath, st->md5); out: SAFE_FREE(suri); SAFE_FREE(duri); SAFE_FREE(tdir); /* set instruction for the statedb merger */ if (rc != 0) { _csync_file_stat_set_error(st, csync_get_error_string(ctx)); if (other) { /* We set the instruction to UPDATED so next try we try to rename again */ st->instruction = CSYNC_INSTRUCTION_UPDATED; _csync_record_error(ctx, other, pi); pi = NULL; } } csync_statedb_free_progressinfo(pi); return rc; } static int _csync_sync_file(CSYNC *ctx, csync_file_stat_t *st) { int rc = -1; rc = _csync_push_file(ctx, st); return rc; } static int _csync_conflict_file(CSYNC *ctx, csync_file_stat_t *st) { int rc = -1; char *conflict_file_name; char *uri = NULL; rc = _csync_backup_file(ctx, st, &conflict_file_name); if(rc>=0) { rc = _csync_push_file(ctx, st); } if( rc >= 0 ) { /* if its the local repository, check if both files are equal. */ if( ctx->current == REMOTE_REPLICA ) { if (asprintf(&uri, "%s/%s", ctx->local.uri, st->path) < 0) { return -1; } if( c_compare_file(uri, conflict_file_name) == 1 ) { /* the files are byte wise equal. The conflict can be erased. */ if (csync_vio_local_unlink(conflict_file_name) < 0) { CSYNC_LOG(CSYNC_LOG_PRIORITY_DEBUG, "REMOVE of csync conflict file %s failed.", conflict_file_name ); } else { CSYNC_LOG(CSYNC_LOG_PRIORITY_DEBUG, "REMOVED csync conflict file %s as files are equal.", conflict_file_name ); } } } } return rc; } static int _csync_remove_file(CSYNC *ctx, csync_file_stat_t *st) { char errbuf[256] = {0}; char *uri = NULL; int rc = -1; csync_progressinfo_t *pi = NULL; #ifdef BLACKLIST_ON_ERROR pi = csync_statedb_get_progressinfo(ctx, st->phash, st->modtime, st->md5); if (pi && pi->error > 3) { CSYNC_LOG(CSYNC_LOG_PRIORITY_ERROR, "'%s' was blacklisted after %d errors: %s", st->path, pi->error, pi->error_string); rc = 1; if (!st->error_string && pi->error_string) st->error_string = c_strdup(pi->error_string); goto out; } #endif switch (ctx->current) { case LOCAL_REPLICA: if (asprintf(&uri, "%s/%s", ctx->local.uri, st->path) < 0) { return -1; } break; case REMOTE_REPLICA: if (_csync_build_remote_uri(ctx, &uri, st->path) < 0) { return -1; } break; default: break; } _notify_progress(ctx, uri, st->size, CSYNC_NOTIFY_START_DELETE); if (csync_vio_unlink(ctx, uri) < 0) { switch (errno) { case ENOMEM: rc = -1; break; default: rc = 1; break; } C_STRERROR(errno, errbuf, sizeof(errbuf)); CSYNC_LOG(CSYNC_LOG_PRIORITY_ERROR, "file: %s, command: unlink, error: %s", uri, errbuf); goto out; } /* set instruction for the statedb merger */ st->instruction = CSYNC_INSTRUCTION_DELETED; _notify_progress(ctx, uri, st->size, CSYNC_NOTIFY_END_DELETE); CSYNC_LOG(CSYNC_LOG_PRIORITY_DEBUG, "REMOVED file: %s", uri); rc = 0; out: SAFE_FREE(uri); /* set instruction for the statedb merger */ if (rc != 0) { /* Write file to statedb, to try to sync again on the next run. */ st->instruction = CSYNC_INSTRUCTION_NONE; _csync_record_error(ctx, st, pi); pi = NULL; } csync_statedb_free_progressinfo(pi); return rc; } static int _csync_new_dir(CSYNC *ctx, csync_file_stat_t *st) { enum csync_replica_e dest = -1; enum csync_replica_e replica_bak; char errbuf[256] = {0}; char *uri = NULL; struct timeval times[2]; int rc = -1; csync_progressinfo_t *pi = NULL; #ifdef BLACKLIST_ON_ERROR pi = csync_statedb_get_progressinfo(ctx, st->phash, st->modtime, st->md5); if (pi && pi->error > 3) { CSYNC_LOG(CSYNC_LOG_PRIORITY_ERROR, "'%s' was blacklisted after %d errors: %s", st->path, pi->error, pi->error_string); if (!st->error_string && pi->error_string) st->error_string = c_strdup(pi->error_string); rc = 1; goto out; } #endif replica_bak = ctx->replica; switch (ctx->current) { case LOCAL_REPLICA: dest = ctx->remote.type; if (_csync_build_remote_uri(ctx, &uri, st->path) < 0) { return -1; } break; case REMOTE_REPLICA: dest = ctx->local.type; if (asprintf(&uri, "%s/%s", ctx->local.uri, st->path) < 0) { return -1; } break; default: break; } ctx->replica = dest; if (csync_vio_mkdirs(ctx, uri, C_DIR_MODE) < 0) { switch (errno) { case ENOMEM: rc = -1; break; default: rc = 1; break; } C_STRERROR(errno, errbuf, sizeof(errbuf)); CSYNC_LOG(CSYNC_LOG_PRIORITY_ERROR, "dir: %s, command: mkdirs, error: %s", uri, errbuf); goto out; } /* chmod is if it is not the default mode */ if ((st->mode & 07777) != C_DIR_MODE) { if (csync_vio_chmod(ctx, uri, st->mode) < 0) { switch (errno) { case ENOMEM: rc = -1; break; default: rc = 1; break; } C_STRERROR(errno, errbuf, sizeof(errbuf)); CSYNC_LOG(CSYNC_LOG_PRIORITY_ERROR, "dir: %s, command: chmod, error: %s", uri, errbuf); goto out; } } /* set owner and group if possible */ if (ctx->pwd.euid == 0) { csync_vio_chown(ctx, uri, st->uid, st->gid); } times[0].tv_sec = times[1].tv_sec = st->modtime; times[0].tv_usec = times[1].tv_usec = 0; csync_vio_utimes(ctx, uri, times); /* set instruction for the statedb merger */ st->instruction = CSYNC_INSTRUCTION_UPDATED; CSYNC_LOG(CSYNC_LOG_PRIORITY_DEBUG, "CREATED dir: %s", uri); ctx->replica = replica_bak; rc = 0; out: SAFE_FREE(uri); /* set instruction for the statedb merger */ if (rc != 0) { _csync_record_error(ctx, st, pi); pi = NULL; } csync_statedb_free_progressinfo(pi); return rc; } static int _csync_sync_dir(CSYNC *ctx, csync_file_stat_t *st) { enum csync_replica_e dest = -1; enum csync_replica_e replica_bak; char errbuf[256] = {0}; char *uri = NULL; struct timeval times[2]; int rc = -1; csync_progressinfo_t *pi = NULL; #ifdef BLACKLIST_ON_ERROR pi = csync_statedb_get_progressinfo(ctx, st->phash, st->modtime, st->md5); if (pi && pi->error > 3) { CSYNC_LOG(CSYNC_LOG_PRIORITY_ERROR, "'%s' was blacklisted after %d errors: %s", st->path, pi->error, pi->error_string); if (!st->error_string && pi->error_string) st->error_string = c_strdup(pi->error_string); rc = 1; goto out; } #endif replica_bak = ctx->replica; switch (ctx->current) { case LOCAL_REPLICA: dest = ctx->remote.type; if (_csync_build_remote_uri(ctx, &uri, st->path) < 0) { return -1; } break; case REMOTE_REPLICA: dest = ctx->local.type; if (asprintf(&uri, "%s/%s", ctx->local.uri, st->path) < 0) { return -1; } break; default: break; } ctx->replica = dest; /* chmod is if it is not the default mode */ if ((st->mode & 07777) != C_DIR_MODE) { if (csync_vio_chmod(ctx, uri, st->mode) < 0) { switch (errno) { case ENOMEM: rc = -1; break; default: rc = 1; break; } C_STRERROR(errno, errbuf, sizeof(errbuf)); CSYNC_LOG(CSYNC_LOG_PRIORITY_ERROR, "dir: %s, command: chmod, error: %s", uri, errbuf); goto out; } } /* set owner and group if possible */ if (ctx->pwd.euid == 0) { csync_vio_chown(ctx, uri, st->uid, st->gid); } times[0].tv_sec = times[1].tv_sec = st->modtime; times[0].tv_usec = times[1].tv_usec = 0; csync_vio_utimes(ctx, uri, times); /* set instruction for the statedb merger */ st->instruction = CSYNC_INSTRUCTION_UPDATED; CSYNC_LOG(CSYNC_LOG_PRIORITY_DEBUG, "SYNCED dir: %s", uri); ctx->replica = replica_bak; rc = 0; out: SAFE_FREE(uri); /* set instruction for the statedb merger */ if (rc != 0) { _csync_record_error(ctx, st, pi); pi = NULL; } csync_statedb_free_progressinfo(pi); return rc; } /* If a remove operation failed, we need to update the st so the information that will be stored in the database will make it so that we try to remove again on the next sync. */ static void _csync_remove_error(CSYNC *ctx, csync_file_stat_t *st, char *uri) { /* Write it back to statedb, that we try to delete it next time. */ st->instruction = CSYNC_INSTRUCTION_NONE; if (ctx->replica == LOCAL_REPLICA) { /* Update the mtime */ csync_vio_file_stat_t* vst = csync_vio_file_stat_new(); if (csync_vio_stat(ctx, uri, vst) == 0) { st->inode = vst->inode; st->modtime = vst->mtime; } csync_vio_file_stat_destroy(vst); /* don't write the md5 to the database */ SAFE_FREE(st->md5); } } static int _csync_remove_dir(CSYNC *ctx, csync_file_stat_t *st) { c_list_t *list = NULL; char errbuf[256] = {0}; char *uri = NULL; int rc = -1; csync_file_stat_t **pst; switch (ctx->current) { case LOCAL_REPLICA: if (asprintf(&uri, "%s/%s", ctx->local.uri, st->path) < 0) { return -1; } break; case REMOTE_REPLICA: if (_csync_build_remote_uri(ctx, &uri, st->path) < 0) { return -1; } break; default: break; } if (csync_vio_rmdir(ctx, uri) < 0) { switch (errno) { case ENOMEM: C_STRERROR(errno, errbuf, sizeof(errbuf)); CSYNC_LOG(CSYNC_LOG_PRIORITY_FATAL, "dir: %s, command: rmdir, error: %s", uri, errbuf); rc = -1; break; case ENOTEMPTY: pst = c_malloc(sizeof(csync_file_stat_t*)); *pst = st; switch (ctx->current) { case LOCAL_REPLICA: list = c_list_prepend(ctx->local.list, (void *) pst); if (list == NULL) { return -1; } ctx->local.list = list; break; case REMOTE_REPLICA: list = c_list_prepend(ctx->remote.list, (void *) pst); if (list == NULL) { return -1; } ctx->remote.list = list; break; default: break; } rc = 0; break; default: C_STRERROR(errno, errbuf, sizeof(errbuf)); CSYNC_LOG(CSYNC_LOG_PRIORITY_ERROR, "dir: %s, command: rmdir, error: %s", uri, errbuf); rc = 1; break; } goto out; } /* set instruction for the statedb merger */ st->instruction = CSYNC_INSTRUCTION_DELETED; CSYNC_LOG(CSYNC_LOG_PRIORITY_DEBUG, "REMOVED dir: %s", uri); rc = 0; out: /* set instruction for the statedb merger */ if (rc != 0) { _csync_remove_error(ctx, st, uri); } SAFE_FREE(uri); return rc; } static int _csync_propagation_cleanup(CSYNC *ctx) { c_list_t *list = NULL; c_list_t *walk = NULL; c_list_t *walk2 = NULL; c_list_t *ignored_cleanup = NULL; char *uri = NULL; char *dir = NULL; switch (ctx->current) { case LOCAL_REPLICA: list = ctx->local.list; ignored_cleanup = ctx->local.ignored_cleanup; uri = ctx->local.uri; break; case REMOTE_REPLICA: list = ctx->remote.list; ignored_cleanup = ctx->remote.ignored_cleanup; uri = ctx->remote.uri; break; default: break; } if (list == NULL) { return 0; } list = c_list_sort(list, _csync_cleanup_cmp); if (list == NULL) { return -1; } for (walk = c_list_last(list); walk != NULL; walk = c_list_prev(walk)) { csync_file_stat_t *st = NULL; csync_file_stat_t **pst = NULL; pst = (csync_file_stat_t **) walk->data; st = *(pst); /* Cleanup ignored files */ for (walk2 = c_list_last(ignored_cleanup); walk2 != NULL; walk2 = c_list_prev(walk2)) { const char *fn = (const char*) walk2->data; /* check if the file name does not starts with the path to remove. */ if (strlen(fn) < st->pathlen || fn[st->pathlen] != '/' || strncmp(fn, st->path, st->pathlen) != 0) { continue; } if (asprintf(&dir, "%s/%s", uri, fn) < 0) { return -1; } CSYNC_LOG(CSYNC_LOG_PRIORITY_DEBUG, "Removing ignored file %s ", dir); if (csync_vio_unlink(ctx, dir) < 0) { return -1; } SAFE_FREE(dir); } if (asprintf(&dir, "%s/%s", uri, st->path) < 0) { return -1; } if (csync_vio_rmdir(ctx, dir) < 0) { _csync_remove_error(ctx, st, uri); } else { st->instruction = CSYNC_INSTRUCTION_DELETED; } CSYNC_LOG(CSYNC_LOG_PRIORITY_DEBUG, "CLEANUP dir: %s", dir); SAFE_FREE(dir); SAFE_FREE(pst); } return 0; } static int _csync_propagation_file_count_visitor(void *obj, void *data) { csync_file_stat_t *st = NULL; CSYNC *ctx = NULL; st = (csync_file_stat_t *) obj; ctx = (CSYNC *) data; if (st == NULL) { return -1; } if (ctx == NULL) { return -1; } switch(st->type) { case CSYNC_FTW_TYPE_SLINK: break; case CSYNC_FTW_TYPE_FILE: switch (st->instruction) { case CSYNC_INSTRUCTION_NEW: case CSYNC_INSTRUCTION_SYNC: case CSYNC_INSTRUCTION_CONFLICT: ctx->overall_progress.file_count++; ctx->overall_progress.byte_sum += st->size; break; default: break; } break; case CSYNC_FTW_TYPE_DIR: /* * No counting of directories. */ break; default: break; } return 0; } static int _csync_propagation_file_visitor(void *obj, void *data) { csync_file_stat_t *st = NULL; CSYNC *ctx = NULL; int rc = 0; st = (csync_file_stat_t *) obj; ctx = (CSYNC *) data; if (ctx->abort) { CSYNC_LOG(CSYNC_LOG_PRIORITY_TRACE, "Aborted!"); ctx->error_code = CSYNC_ERR_ABORTED; return -1; } switch(st->type) { case CSYNC_FTW_TYPE_SLINK: break; case CSYNC_FTW_TYPE_FILE: switch (st->instruction) { case CSYNC_INSTRUCTION_NEW: if ((rc = _csync_new_file(ctx, st)) < 0) { CSYNC_LOG(CSYNC_LOG_PRIORITY_TRACE,"FAIL NEW: %s",st->path); goto err; } break; case CSYNC_INSTRUCTION_SYNC: if ((rc = _csync_sync_file(ctx, st)) < 0) { CSYNC_LOG(CSYNC_LOG_PRIORITY_TRACE,"FAIL SYNC: %s",st->path); goto err; } break; case CSYNC_INSTRUCTION_REMOVE: if ((rc = _csync_remove_file(ctx, st)) < 0) { CSYNC_LOG(CSYNC_LOG_PRIORITY_TRACE,"FAIL REMOVE: %s",st->path); goto err; } break; case CSYNC_INSTRUCTION_CONFLICT: CSYNC_LOG(CSYNC_LOG_PRIORITY_TRACE,"case CSYNC_INSTRUCTION_CONFLICT: %s",st->path); if ((rc = _csync_conflict_file(ctx, st)) < 0) { goto err; } break; default: break; } break; case CSYNC_FTW_TYPE_DIR: /* * We have to walk over the files first. If you create or rename a file * in a directory on unix. The modification time of the directory gets * changed. */ break; default: break; } return rc; err: return -1; } static int _csync_propagation_dir_visitor(void *obj, void *data) { csync_file_stat_t *st = NULL; CSYNC *ctx = NULL; st = (csync_file_stat_t *) obj; ctx = (CSYNC *) data; if (ctx->abort) { CSYNC_LOG(CSYNC_LOG_PRIORITY_TRACE, "Aborted!"); ctx->error_code = CSYNC_ERR_ABORTED; return -1; } switch(st->type) { case CSYNC_FTW_TYPE_SLINK: /* FIXME: implement symlink support */ break; case CSYNC_FTW_TYPE_FILE: break; case CSYNC_FTW_TYPE_DIR: switch (st->instruction) { case CSYNC_INSTRUCTION_NEW: if (_csync_new_dir(ctx, st) < 0) { goto err; } break; case CSYNC_INSTRUCTION_SYNC: if (_csync_sync_dir(ctx, st) < 0) { goto err; } break; case CSYNC_INSTRUCTION_CONFLICT: CSYNC_LOG(CSYNC_LOG_PRIORITY_TRACE,"directory attributes different"); if (_csync_sync_dir(ctx, st) < 0) { goto err; } break; case CSYNC_INSTRUCTION_REMOVE: if (_csync_remove_dir(ctx, st) < 0) { goto err; } break; case CSYNC_INSTRUCTION_RENAME: /* there can't be a rename for dirs. See updater. */ break; default: break; } break; default: break; } return 0; err: return -1; } int csync_propagate_rename_file(CSYNC *ctx, csync_file_stat_t *st) { return _csync_rename_file(ctx, st); } /* Count the files to transmit for both up- and download, ie. in both replicas. */ int csync_init_overall_progress(CSYNC *ctx) { if (ctx == NULL) { return -1; } if (ctx->callbacks.progress_cb == NULL) { return 0; } ctx->current = REMOTE_REPLICA; ctx->replica = ctx->remote.type; if (c_rbtree_walk(ctx->remote.tree, (void *) ctx, _csync_propagation_file_count_visitor) < 0) { ctx->error_code = CSYNC_ERR_TREE; return -1; } ctx->current = LOCAL_REPLICA; ctx->replica = ctx->local.type; if (c_rbtree_walk(ctx->local.tree, (void *) ctx, _csync_propagation_file_count_visitor) < 0) { ctx->error_code = CSYNC_ERR_TREE; return -1; } /* Notify the progress */ csync_set_module_property(ctx, "overall_progress_data", &(ctx->overall_progress)); if (ctx->overall_progress.file_count >0) { _notify_progress(ctx, NULL, 0, CSYNC_NOTIFY_START_SYNC_SEQUENCE); } return 0; } void csync_finalize_progress(CSYNC *ctx) { if (ctx->overall_progress.file_count >0) { _notify_progress(ctx, NULL, 0, CSYNC_NOTIFY_FINISHED_SYNC_SEQUENCE); } csync_set_module_property(ctx, "overall_progress_data", NULL); } int csync_propagate_files(CSYNC *ctx) { c_rbtree_t *tree = NULL; switch (ctx->current) { case LOCAL_REPLICA: tree = ctx->local.tree; break; case REMOTE_REPLICA: tree = ctx->remote.tree; break; default: break; } if (c_rbtree_walk(tree, (void *) ctx, _csync_propagation_file_visitor) < 0) { return -1; } if (c_rbtree_walk(tree, (void *) ctx, _csync_propagation_dir_visitor) < 0) { return -1; } if (_csync_propagation_cleanup(ctx) < 0) { return -1; } return 0; } /* vim: set ts=8 sw=2 et cindent: */