#include "support.h" #include "csync_update.c" CSYNC *csync; static void setup(void) { fail_if(system("mkdir -p /tmp/check_csync") < 0, "Setup failed"); fail_if(system("mkdir -p /tmp/check_csync1") < 0, "Setup failed"); fail_if(system("mkdir -p /tmp/check_csync2") < 0, "Setup failed"); fail_if(csync_create(&csync, "/tmp/check_csync1", "/tmp/check_csync2") < 0, "Setup failed"); fail_if(csync_set_config_dir(csync, "/tmp/check_csync") < 0, "Setup failed"); fail_if(csync_init(csync) < 0, NULL); } static void setup_ftw(void) { fail_if(system("mkdir -p /tmp/check_csync") < 0, "Setup failed"); fail_if(system("mkdir -p /tmp/check_csync1") < 0, "Setup failed"); fail_if(system("mkdir -p /tmp/check_csync2") < 0, "Setup failed"); fail_if(csync_create(&csync, "/tmp", "/tmp") < 0, "Setup failed"); fail_if(csync_set_config_dir(csync, "/tmp/check_csync") < 0, "Setup failed"); fail_if(csync_init(csync) < 0, NULL); } static void teardown(void) { fail_if(csync_destroy(csync) < 0, "Teardown failed"); unsetenv("CSYNC_NOMEMORY"); } static void teardown_rm(void) { teardown(); fail_if(system("rm -rf /tmp/check_csync") < 0, "Teardown failed"); fail_if(system("rm -rf /tmp/check_csync1") < 0, "Teardown failed"); fail_if(system("rm -rf /tmp/check_csync2") < 0, "Teardown failed"); } /* create a file stat, caller must free memory */ static csync_vio_file_stat_t* create_fstat(const char *name, ino_t inode, nlink_t nlink, time_t mtime) { csync_vio_file_stat_t *fs = NULL; time_t t; fs = csync_vio_file_stat_new(); if (fs == NULL) { return NULL; } if (name && *name) { fs->name = c_strdup(name); } else { fs->name = c_strdup("file.txt"); } if (fs->name == NULL) { csync_vio_file_stat_destroy(fs); return NULL; } fs->fields = CSYNC_VIO_FILE_STAT_FIELDS_NONE; fs->type = CSYNC_VIO_FILE_TYPE_REGULAR; fs->fields |= CSYNC_VIO_FILE_STAT_FIELDS_TYPE; fs->mode = 0644; fs->fields |= CSYNC_VIO_FILE_STAT_FIELDS_PERMISSIONS; if (inode == 0) { fs->inode = 619070; } else { fs->inode = inode; } fs->fields |= CSYNC_VIO_FILE_STAT_FIELDS_INODE; fs->device = 0; fs->size = 157459; fs->fields |= CSYNC_VIO_FILE_STAT_FIELDS_SIZE; if (nlink == 0) { fs->nlink = 1; } else { fs->nlink = nlink; } fs->fields |= CSYNC_VIO_FILE_STAT_FIELDS_LINK_COUNT; fs->uid = 1000; fs->fields |= CSYNC_VIO_FILE_STAT_FIELDS_UID; fs->gid = 1000; fs->fields |= CSYNC_VIO_FILE_STAT_FIELDS_GID; fs->blkcount = 312; fs->fields |= CSYNC_VIO_FILE_STAT_FIELDS_BLOCK_COUNT; fs->blksize = 4096; fs->fields |= CSYNC_VIO_FILE_STAT_FIELDS_BLOCK_SIZE; if (mtime == 0) { fs->atime = fs->ctime = fs->mtime = time(&t); } else { fs->atime = fs->ctime = fs->mtime = mtime; } fs->fields |= CSYNC_VIO_FILE_STAT_FIELDS_ATIME; fs->fields |= CSYNC_VIO_FILE_STAT_FIELDS_CTIME; fs->fields |= CSYNC_VIO_FILE_STAT_FIELDS_MTIME; return fs; } static int failing_fn(CSYNC *ctx, const char *file, const csync_vio_file_stat_t *fs, enum csync_ftw_flags_e flag) { (void) ctx; (void) file; (void) fs; (void) flag; return -1; } /* detect a new file */ START_TEST (check_csync_detect_update) { csync_file_stat_t *st = NULL; csync_vio_file_stat_t *fs = NULL; fs = create_fstat("file.txt", 0, 1, 1217597845); fail_unless(_csync_detect_update(csync, "/tmp/check_csync1/file.txt", fs, CSYNC_FTW_TYPE_FILE) == 0, NULL); /* the instruction should be set to new */ st = c_rbtree_node_data(csync->local.tree->root); fail_unless(st->instruction == CSYNC_INSTRUCTION_NEW, "instruction is %s", csync_instruction_str(st->instruction)); /* set the instruction to UPDATED that it gets written to the statedb */ st->instruction = CSYNC_INSTRUCTION_UPDATED; /* create a statedb */ csync_set_status(csync, 0xFFFF); csync_vio_file_stat_destroy(fs); } END_TEST START_TEST (check_csync_detect_update_db_none) { csync_file_stat_t *st = NULL; csync_vio_file_stat_t *fs = NULL; fs = create_fstat("file.txt", 0, 1, 1217597845); fail_unless(_csync_detect_update(csync, "/tmp/check_csync1/file.txt", fs, CSYNC_FTW_TYPE_FILE) == 0, NULL); /* the instruction should be set to new */ st = c_rbtree_node_data(csync->local.tree->root); fail_unless(st->instruction == CSYNC_INSTRUCTION_NONE, "instruction is %s", csync_instruction_str(st->instruction)); /* set the instruction to UPDATED that it gets written to the statedb */ st->instruction = CSYNC_INSTRUCTION_UPDATED; /* create a statedb */ csync_set_status(csync, 0xFFFF); csync_vio_file_stat_destroy(fs); } END_TEST START_TEST (check_csync_detect_update_db_eval) { csync_file_stat_t *st = NULL; csync_vio_file_stat_t *fs = NULL; fs = create_fstat("file.txt", 0, 1, 0); fail_unless(_csync_detect_update(csync, "/tmp/check_csync1/file.txt", fs, CSYNC_FTW_TYPE_FILE) == 0, NULL); /* the instruction should be set to new */ st = c_rbtree_node_data(csync->local.tree->root); fail_unless(st->instruction == CSYNC_INSTRUCTION_EVAL, "instruction is %s", csync_instruction_str(st->instruction)); /* set the instruction to UPDATED that it gets written to the statedb */ st->instruction = CSYNC_INSTRUCTION_UPDATED; /* create a statedb */ csync_set_status(csync, 0xFFFF); csync_vio_file_stat_destroy(fs); } END_TEST START_TEST (check_csync_detect_update_db_rename) { csync_file_stat_t *st = NULL; csync_vio_file_stat_t *fs = NULL; fs = create_fstat("wurst.txt", 0, 1, 0); fail_unless(_csync_detect_update(csync, "/tmp/check_csync1/wurst.txt", fs, CSYNC_FTW_TYPE_FILE) == 0, NULL); /* the instruction should be set to rename */ st = c_rbtree_node_data(csync->local.tree->root); fail_unless(st->instruction == CSYNC_INSTRUCTION_RENAME, "instruction is %s", csync_instruction_str(st->instruction)); /* set the instruction to UPDATED that it gets written to the statedb */ st->instruction = CSYNC_INSTRUCTION_UPDATED; /* create a statedb */ csync_set_status(csync, 0xFFFF); csync_vio_file_stat_destroy(fs); } END_TEST START_TEST (check_csync_detect_update_db_new) { csync_file_stat_t *st = NULL; csync_vio_file_stat_t *fs = NULL; fs = create_fstat("file.txt", 42000, 1, 0); fail_unless(_csync_detect_update(csync, "/tmp/check_csync1/file.txt", fs, CSYNC_FTW_TYPE_FILE) == 0, NULL); /* the instruction should be set to new */ st = c_rbtree_node_data(csync->local.tree->root); fail_unless(st->instruction == CSYNC_INSTRUCTION_NEW, "instruction is %s", csync_instruction_str(st->instruction)); /* set the instruction to UPDATED that it gets written to the statedb */ st->instruction = CSYNC_INSTRUCTION_UPDATED; /* create a statedb */ csync_set_status(csync, 0xFFFF); csync_vio_file_stat_destroy(fs); } END_TEST START_TEST (check_csync_detect_update_nlink) { csync_file_stat_t *st = NULL; csync_vio_file_stat_t *fs = NULL; /* create vio file stat with nlink greater than 1 */ fs = create_fstat("file.txt", 0, 7, 0); /* add it to local tree */ fail_unless(_csync_detect_update(csync, "/tmp/check_csync1/file.txt", fs, CSYNC_FTW_TYPE_FILE) == 0, NULL); /* the instruction should be set to ignore */ st = c_rbtree_node_data(csync->local.tree->root); fail_unless(st->instruction == CSYNC_INSTRUCTION_IGNORE); csync_vio_file_stat_destroy(fs); } END_TEST START_TEST (check_csync_detect_update_null) { csync_vio_file_stat_t *fs = NULL; fs = create_fstat("file.txt", 0, 1, 0); fail_unless(_csync_detect_update(csync, NULL, fs, CSYNC_FTW_TYPE_FILE) < 0, NULL); fail_unless(_csync_detect_update(csync, "/tmp/check_csync1/file.txt", NULL, CSYNC_FTW_TYPE_FILE) < 0, NULL); csync_vio_file_stat_destroy(fs); } END_TEST START_TEST (check_csync_ftw) { fail_unless(csync_ftw(csync, "/tmp", csync_walker, MAX_DEPTH) == 0, NULL); } END_TEST START_TEST (check_csync_ftw_empty_uri) { fail_unless(csync_ftw(csync, "", csync_walker, MAX_DEPTH) < 0, NULL); } END_TEST START_TEST (check_csync_ftw_failing_fn) { fail_unless(csync_ftw(csync, "/tmp", failing_fn, MAX_DEPTH) < 0, NULL); } END_TEST #ifdef CSYNC_MEM_NULL_TESTS START_TEST (check_csync_detect_update_no_mem) { csync_vio_file_stat_t *fs = NULL; fs = create_fstat("file.txt", 0, 1, 0); setenv("CSYNC_NOMEMORY", "1", 1); fail_unless(_csync_detect_update(csync, "file.txt", fs, CSYNC_FTW_TYPE_FILE) < 0, NULL); csync_vio_file_stat_destroy(fs); } END_TEST START_TEST (check_csync_ftw_nomem) { setenv("CSYNC_NOMEMORY", "1", 1); fail_unless(csync_ftw(csync, "/tmp", csync_walker, MAX_DEPTH) < 0, NULL); } END_TEST #endif static Suite *make_csync_suite(void) { Suite *s = suite_create("csync_update"); create_case_fixture(s, "check_csync_detect_update", check_csync_detect_update, setup, teardown); create_case_fixture(s, "check_csync_detect_update_db_none", check_csync_detect_update_db_none, setup, teardown); create_case_fixture(s, "check_csync_detect_update_db_eval", check_csync_detect_update_db_eval, setup, teardown); create_case_fixture(s, "check_csync_detect_update_db_rename", check_csync_detect_update_db_rename, setup, teardown); create_case_fixture(s, "check_csync_detect_update_db_new", check_csync_detect_update_db_new, setup, teardown_rm); create_case_fixture(s, "check_csync_detect_update_nlink", check_csync_detect_update_nlink, setup, teardown_rm); create_case_fixture(s, "check_csync_detect_update_no_file", check_csync_detect_update_null, setup, teardown_rm); create_case_fixture(s, "check_csync_ftw", check_csync_ftw, setup_ftw, teardown_rm); create_case_fixture(s, "check_csync_ftw_empty_uri", check_csync_ftw_empty_uri, setup_ftw, teardown_rm); create_case_fixture(s, "check_csync_ftw_failing_fn", check_csync_ftw_failing_fn, setup, teardown_rm); #ifdef CSYNC_MEM_NULL_TESTS create_case_fixture(s, "check_csync_ftw_nomem", check_csync_ftw_nomem, setup, teardown_rm); create_case_fixture(s, "check_csync_detect_update_nomem", check_csync_detect_update_no_mem, setup, teardown_rm); #endif return s; } int main(int argc, char **argv) { Suite *s = NULL; SRunner *sr = NULL; struct argument_s arguments; int nf; ZERO_STRUCT(arguments); cmdline_parse(argc, argv, &arguments); s = make_csync_suite(); sr = srunner_create(s); if (arguments.nofork) { srunner_set_fork_status(sr, CK_NOFORK); } srunner_run_all(sr, CK_VERBOSE); nf = srunner_ntests_failed(sr); srunner_free(sr); return (nf == 0) ? EXIT_SUCCESS : EXIT_FAILURE; }