summaryrefslogtreecommitdiff
path: root/src
diff options
context:
space:
mode:
Diffstat (limited to 'src')
-rw-r--r--src/commit.c1575
1 files changed, 0 insertions, 1575 deletions
diff --git a/src/commit.c b/src/commit.c
deleted file mode 100644
index c6080fe..0000000
--- a/src/commit.c
+++ /dev/null
@@ -1,1575 +0,0 @@
-#include <stdlib.h>
-#include <stdio.h>
-#include <string.h>
-#include <assert.h>
-#include <ctype.h>
-#include <errno.h>
-#include <sys/types.h>
-#include <sys/stat.h>
-#include <unistd.h>
-#include <fcntl.h>
-#include <dirent.h>
-#include <limits.h>
-
-#include "cli_val.h"
-#include "cli_objects.h"
-#include "cli_parse.h"
-#include "cli_path_utils.h"
-
-struct DirIndex {
- int dirname_index;
- int dirname_ct;
- struct DirSort** dirname;
-};
-
-struct DirSort {
- unsigned long priority;
- char name[255];
-};
-
-
-static const char def_name[] = DEF_NAME;
-static const char tag_name[] = TAG_NAME;
-static const char opaque_name[] = OPQ_NAME;
-
-static int fin_commit(boolean ok);
-static boolean commit_value(vtw_def *defp, char *cp,
- vtw_cmode mode, boolean in_txn);
-static void perform_create_node(void);
-static void perform_delete_node(void);
-static void perform_move(void);
-static boolean commit_delete_child(vtw_def *pdefp, char *child,
- boolean deleting, boolean in_txn);
-static boolean commit_delete_children(vtw_def *defp, boolean deleting,
- boolean in_txn);
-static boolean commit_update_children(vtw_def *defp, boolean creating,
- boolean in_txn, boolean *parent_update);
-
-extern char *cli_operation_name;
-
-#if BITWISE
-static void make_dir(void)
-{
- struct stat statbuf;
- if (lstat(m_path.path, &statbuf) < 0) {
- mkdir_p(m_path.path);
- return;
- }
- if (!S_ISDIR(statbuf.st_mode)) {
- bye("directory %s expected, found file", m_path.path);
- }
-}
-#endif
-
-static int
-compare_dirname(const void *p, const void *q)
-{
- const struct DirSort *a = (const struct DirSort*)*(struct Dirsort **)p;
- const struct DirSort *b = (const struct DirSort*)*(struct Dirsort **)q;
- if (a->priority == b->priority) {
- return strcmp(a->name,b->name);
- }
- return ((long)b->priority - (long)a->priority);
-}
-
-static int
-compare_dirname_reverse_priority(const void *p, const void *q)
-{
- const struct DirSort *a = (const struct DirSort*)*(struct Dirsort **)p;
- const struct DirSort *b = (const struct DirSort*)*(struct Dirsort **)q;
- if (a->priority == b->priority) {
- return strcmp(a->name,b->name);
- }
- return ((long)a->priority - (long)b->priority);
-}
-
-static struct DirIndex*
-init_next_filtered_dirname(DIR *dp, int sort_order, int exclude_wh)
-{
- struct DirIndex *di = malloc(sizeof(struct DirIndex));
- di->dirname = malloc(1024 * sizeof(char*));
- di->dirname_ct = 0;
- di->dirname_index = 0;
-
- struct dirent *dirp;
- while ((dirp = readdir(dp)) != NULL) {
- if (strcmp(dirp->d_name, ".") == 0 || strcmp(dirp->d_name, "..") == 0
- || strcmp(dirp->d_name, MOD_NAME) == 0
- || strcmp(dirp->d_name, opaque_name) == 0
- || (exclude_wh && strncmp(dirp->d_name, ".wh.", 4) == 0)) {
- continue;
- }
- else {
- struct DirSort *d = malloc(sizeof(struct DirSort));
- if (strlen(dirp->d_name) >= 255) {
- bye("configuration value exceeds 255 chars\n");
- }
- strcpy(d->name,dirp->d_name);
- d->priority = (unsigned long)0;
- vtw_def def;
-
- char path[strlen(t_path.path)+strlen(dirp->d_name)+2+8+1];
- sprintf(path,"%s/%s/node.def", t_path.path, dirp->d_name);
-
- struct stat s;
- if ((lstat(path,&s) == 0) && S_ISREG(s.st_mode)) {
- memset(&def, 0, sizeof(def));
- if (parse_def(&def,path,FALSE) == 0) {
- d->priority = def.def_priority;
- }
- }
-
- di->dirname[di->dirname_ct++] = d;
- if (di->dirname_ct % 1024 == 0) {
- di->dirname = realloc(di->dirname, (di->dirname_ct+1024)*sizeof(char*));
- }
- }
- }
- if (sort_order == 0) {
- qsort(di->dirname, di->dirname_ct, sizeof(char*), compare_dirname);
- }
- else {
- qsort(di->dirname, di->dirname_ct, sizeof(char*), compare_dirname_reverse_priority);
- }
- return di;
-}
-
-static char*
-get_next_filtered_dirname(struct DirIndex *di)
-{
- if (di == NULL || di->dirname_index == di->dirname_ct) {
- return NULL;
- }
- return di->dirname[di->dirname_index++]->name;
-}
-
-static void
-release_dir_index(struct DirIndex *di)
-{
- if (di != NULL) {
- int i;
- for (i = 0; i < di->dirname_ct; ++i) {
- if (di->dirname[i] != NULL) {
- free((struct DirSort*)di->dirname[i]);
- }
- }
- free(di);
- di = NULL;
- }
-}
-
-/*************************************************
- validate_dir_for_commit:
- validate value.value if there is one, validate
- subdirectries names (for tag directory, and for
- regular directory);
- validate subdirectories
- returns TRUE if OK, FASLE if errors
- exits with status != 0 in case of parse error
-*/
-static boolean validate_dir_for_commit(void)
-{
- struct stat statbuf;
- int status=0;
- vtw_def def;
- boolean def_present=FALSE;
- boolean value_present=FALSE;
- int subdirs_number=0;
- DIR *dp=NULL;
- char *dirname = NULL;
- char *cp=NULL;
- boolean ret=TRUE;
- char *uename = NULL;
-
-#ifdef DEBUG
- printf("validating directory (node_cnt %d, free_node_cnt %d)\n"
- "t_path |%s|\n",
- node_cnt, free_node_cnt, t_path.path);
-#endif
-
- /* find definition */
-
- push_path(&t_path, def_name); /* PUSH 1 */
- if ((lstat(t_path.path, &statbuf) >= 0) &&
- ((statbuf.st_mode & S_IFMT) == S_IFREG)) {
- /* definition present */
- def_present = TRUE;
- memset(&def, 0, sizeof(def));
-#ifdef DEBUG1
- printf("Parsing definition %s\n", t_path.path);
-#endif
-
- if ((status = parse_def(&def, t_path.path,
- FALSE))) {
- bye("parse error: %d\n", status);
- }
-
- }
-#ifdef DEBUG1
- else
- printf("No definition %s\n", t_path.path);
-#endif
- pop_path(&t_path); /* for PUSH 1 */
-
- /* look at modified stuff */
- if ((dp = opendir(m_path.path)) == NULL){
- INTERNAL;
- }
- if (def_present && def.tag) {
- push_path(&t_path, tag_name); /* PUSH 2a */
- }
- struct DirIndex *di = init_next_filtered_dirname(dp,0,1);
- while ((dirname = get_next_filtered_dirname(di)) != NULL) {
- subdirs_number++;
-
- if(uename)
- my_free(uename);
-
- uename = clind_unescape(dirname);
-
- if (strcmp(uename, VAL_NAME) == 0) {
-
- value_present=TRUE;
-
- /* deal with the value */
- if (!def_present) {
- printf("There is no definition specified in template\n"
- "\t%s\n\t - therefore no value permitted\n",
- t_path.path);
- ret = FALSE;
- continue;
- }
-
- if (def.tag) {
- printf("Tag specified in template\n"
- "\t%s\n\t - therefore no value permitted\n",
- t_path.path);
- ret = FALSE;
- continue;
- }
-
- /* value is OK */
- /* read it */
- cp = NULL;
- status = get_value(&cp, &m_path);
- if (status == VTWERR_OK) {
- size_t value_size = 0;
-#ifdef DEBUG1
- printf("Validating value |%s|\n"
- "for path %s\n", cp, m_path.path);
-#endif
-
- // multivalue has to be diff-ed with the pre-existing value.
- // only the difference should be ran through the validation.
- // Bug #2509
- if (def.multi) {
- char *old_value = NULL;
- value_size = strlen(cp);
- switch_path(APATH); // temporarily switching to the active config path
- status = get_value(&old_value, &m_path);
- if (status == VTWERR_OK) {
- subtract_values(&cp, old_value);
- }
- switch_path(CPATH);
- if (old_value)
- my_free(old_value);
- }
- if (strlen(cp) > 0 || value_size == 0)
- status = validate_value(&def, cp);
- else
- status = TRUE;
- ret = ret && status;
- }
- if (cp)
- my_free(cp);
- continue;
- }
-
- if (strcmp(uename,"def") == 0) {
- ret = TRUE;
- continue;
- }
-
- push_path(&m_path, uename); /* PUSH 3 */
- if (lstat(m_path.path, &statbuf) < 0) {
- printf("Can't read directory %s\n",
- m_path.path);
- ret = FALSE;
- pop_path(&m_path); /* for PUSH 3 */
- continue;
- }
-
- if ((statbuf.st_mode & S_IFMT) != S_IFDIR) {
- printf("Non directory file %s\n", m_path.path);
- ret = FALSE;
- pop_path(&m_path); /* for PUSH 3 */
- continue;
- }
-
- if (def_present && def.tag) {
- /* do not push t_path, it is already pushed above */
- /* validate dir name against definition */
- boolean res = validate_value(&def, uename);
- value_present=TRUE;
- if (!res) {
- ret = FALSE;
- /* do not go inside bad directory */
- pop_path(&m_path); /* for PUSH 3 */
- continue;
- }
- } else {
- push_path(&t_path, uename); /* PUSH 2b the same as PUSH 2a */
- if (lstat(t_path.path, &statbuf) < 0) {
- printf("No such template directory (%s)\n"
- "for directory %s",
- t_path.path, m_path.path);
- ret = FALSE;
- pop_path(&t_path); /* for PUSH 2b */
- pop_path(&m_path); /* for PUSH 3 */
- continue;
- }
- if ((statbuf.st_mode & S_IFMT) != S_IFDIR) {
- printf("Non directory file %s\n", m_path.path);
- ret = FALSE;
- pop_path(&t_path); /* for PUSH 2b */
- pop_path(&m_path); /* for PUSH 3 */
- continue;
- }
- }
-
- status = validate_dir_for_commit();
- ret = ret && status;
- pop_path(&m_path); /* for PUSH 3 */
- if (!def_present || !def.tag)
- pop_path(&t_path); /* for PUSH 2b */
-
- } // while
- release_dir_index(di);
- status = closedir(dp);
- if (status)
- bye("Cannot close dir %s\n", m_path.path);
-
- if(!value_present && def_present && !def.tag) {
- ret = ret && validate_value(&def, "");
- }
-
- if (def_present && def.tag)
- pop_path(&t_path); /* for PUSH 2a */
- if (def_present)
- free_def(&def);
-#ifdef DEBUG
- printf("directory done(node_cnt %d, free_node_cnt %d)\n"
- "mpath |%s|\n",
- node_cnt, free_node_cnt, t_path.path);
-#endif
- if (uename)
- my_free(uename);
- return ret;
-}
-
-/***************************************************
- main:
- main function (for now)
-***************************************************/
-
-int main(int argc, char **argv)
-{
-
- boolean status;
- struct stat statbuf;
- boolean update_pending = FALSE;
-
- cli_operation_name = "Commit";
-
- if (initialize_output() == -1) {
- bye("can't initialize output\n");
- }
-
- set_in_commit(TRUE);
- dump_log( argc, argv);
- init_paths(TRUE);
-
- char mod[strlen(get_mdirp()) + sizeof(MOD_NAME)+2];
- sprintf(mod, "%s/%s", get_mdirp(), MOD_NAME);
- if (lstat(mod, &statbuf) < 0) {
- fprintf(out_stream, "No configuration changes to commit\n");
- return 0;
- }
-
- if (get_config_lock() == -1) {
- fprintf(out_stream, "Cannot commit: Configuration is locked\n");
- bye("Configuration is locked\n");
- }
-
- status = validate_dir_for_commit();
- if (status == TRUE) {
- switch_path(CPATH);
- status = commit_delete_children(NULL, FALSE, FALSE);
- }
- if (status == TRUE)
- status = commit_update_children(NULL, FALSE, FALSE, &update_pending);
- fin_commit(status);
-
- done();
-
- return (status == TRUE) ? 0 : 1;
-}
-
-/*************************************************************
- perform_create_node -
- remove node and descendent from woring path
- and create a new node
-*************************************************************/
-static void perform_create_node(void)
-{
-#if BITWISE
- static const char format[]="rm -f -r %s;mkdir %s";
- char command[2 * strlen(m_path.path) + sizeof(format)];
-
- switch_path(APATH);
- sprintf(command, format, m_path.path, m_path.path);
- system(command);
- switch_path(CPATH);
-#endif
-}
-
-
-/*************************************************************
- perform_delete_node -
- delete node in current path
-*************************************************************/
-static void perform_delete_node(void)
-{
-#if BITWISE
- static const char format[]="rm -f -r %s";
- char command[strlen(m_path.path) + sizeof(format)];
- sprintf(command, format, m_path.path);
- system(command);
-#endif
-}
-
-static void perform_move(void)
-{
-#if BITWISE
- static const char format[] = "rm -r -f %s;mkdir %s;mv %s/" VAL_NAME " %s";
- char *a_path;
- char command[sizeof(format)+3*strlen(a_path)+strlen(m_path.path)];
-
- switch_path(APATH);
- a_path = my_strdup(m_path.path, "");
- switch_path(CPATH);
- sprintf(command, format, a_path, a_path, m_path.path, a_path);
- system(command);
- my_free(a_path);
-#endif
-}
-
-/*************************************************
- commit_update_child:
- preconditions: t_path and c_path set up for parent
- pdefp (IN) - parent definition (may be NULL)
- child (IN) - name of the child
- creating (IN) - mode of commiting (update or create)
- update_parent (OUT) - unfulfilled update request
- commit child and its descendants
- returns TRUE if no errors
- exit if error during execution
-*/
-boolean commit_update_child(vtw_def *pdefp, char *child,
- boolean creating, boolean in_txn, boolean *update_parent)
-{
- boolean update_pending = FALSE;
- boolean multi_tag = FALSE;
- struct stat statbuf;
- int status;
- vtw_def def;
- vtw_def *my_defp; /*def to be given to my children */
- vtw_def *act_defp;/*def to be used for actions */
- char *cp;
- vtw_mark mark;
- vtw_act_type act;
- boolean do_end, ok, do_begin = FALSE, do_txn = FALSE;
-
- set_at_string(NULL);
-
- ok = TRUE;
-#ifdef DEBUG
- printf("commiting directory (node_cnt %d, free_node_cnt %d)\n"
- "mpath |%s|\n",
- node_cnt, free_node_cnt, t_path.path);
-#endif
- if (strcmp(child, ".") == 0 ||
- strcmp(child, "..") == 0 ||
- strcmp(child, MOD_NAME) == 0 ||
- strcmp(child, opaque_name) == 0)
- return TRUE;
-
- /* deleted subdirectory */
- if (strncmp(child, ".wh.", 4) == 0) { /* whiteout */
- /* ignore */
- return TRUE;
- }
- mark_paths(&mark);
- if (!creating) {
- /* are we marked with opaque */
- push_path(&m_path, child);
- push_path(&m_path, opaque_name);
- if (lstat(m_path.path, &statbuf) >= 0 && !creating) {
- creating = TRUE;
- }
- pop_path(&m_path);
- pop_path(&m_path);
- }
- /* find our definition */
- if (pdefp && pdefp->tag) {
- my_defp = NULL;
- act_defp = pdefp;
- push_path(&t_path,tag_name);
- } else {
- push_path(&t_path, child);
- push_path(&t_path, def_name);
- if ((lstat(t_path.path, &statbuf) >= 0) &&
- ((statbuf.st_mode & S_IFMT) == S_IFREG)) {
- /* defniition present */
- act_defp = my_defp = &def;
- memset(&def, 0, sizeof(def));
-#ifdef DEBUG1
- printf("Parsing definition\n");
-#endif
- if ((status = parse_def(&def, t_path.path, FALSE))) {
- bye("parse error: %d\n", status);
- }
- my_defp = act_defp = &def;
- if (def.tag)
- multi_tag = TRUE;
- } else {
- my_defp = act_defp = NULL;
- }
- pop_path(&t_path); /*def_name */
- }
- do_end = FALSE;
- if (!multi_tag && !in_txn && act_defp &&
- (act_defp->actions[begin_act].vtw_list_head ||
- act_defp->actions[end_act].vtw_list_head)){
- /* we are traversing change directory
- if we are here, there is a change somewhere */
- do_txn = in_txn = TRUE;
- if (act_defp->actions[end_act].vtw_list_head)
- do_end = TRUE;
- /* if creating, delete skipped this directory,
- we have to do begin act */
- if (creating &&
- act_defp->actions[begin_act].vtw_list_head)
- do_begin = TRUE;
- }
- push_path(&m_path, child);
- /* are we a "value" parent node */
- if (act_defp && !act_defp->tag && act_defp->def_type != ERROR_TYPE ) {
- /* we are value node */
- if (!act_defp->multi) {
- /* single node */
- /* create_mode do create,
- update_mode do create or update */
-
- if (creating)
- act = create_act;
- else
- act = update_act;
-
- if ((act==update_act) && !act_defp->actions[update_act].vtw_list_head){
- /* updating but no action
- ask parent to do - propagate up */
- *update_parent = TRUE;
- }
- /* look for actions */
- /* if act != create_act => run actions[act] if not empty */
- /* if act == create_act
- * if actions[create_act] not empty
- * run it
- * else
- * run actions[update_act] if not empty
- * run actions[activate_act] if not empty
- */
- if (act_defp->actions[act].vtw_list_head
- || (act == create_act
- && (act_defp->actions[activate_act].vtw_list_head
- || act_defp->actions[update_act].vtw_list_head))) {
- status = get_value_to_at_string(&m_path);
- if (status != VTWERR_OK)
- bye("Can not read value at %s", m_path.path);
- /* remove \n at the line end */
- cp = strchr(get_at_string(), '\n');
- if (cp)
- *cp = 0;
- if (act_defp->actions[act].vtw_list_head) {
- status = execute_list(act_defp->actions[act].vtw_list_head,
- act_defp);
- if (!status) {
-#ifdef DEBUG
- bye("begin action not NULL for %s\n", get_at_string());
-#else
- return(FALSE);
-#endif
- }
- } else {
- /* creating but no create action */
- /* try update action */
- if ((act == create_act)
- && act_defp->actions[update_act].vtw_list_head) {
- status
- = execute_list(act_defp->actions[update_act].vtw_list_head,
- act_defp);
- if (!status) {
- return (FALSE);
- }
- }
- }
- /* try activate action if creating */
- if ((act == create_act)
- && act_defp->actions[activate_act].vtw_list_head) {
- status
- = execute_list(act_defp->actions[activate_act].vtw_list_head,
- act_defp);
- if (!status) {
- return (FALSE);
- }
- }
- free_at_string();
- }
- /* now handle commit value */
- if (!in_txn) { /* ELSE WAIT TILL THE END OF TXN */
- perform_move();
- perform_delete_node();
- }
- goto restore;
- }
- /* else multi_node */
- cp = NULL;
- status = get_value(&cp, &m_path);
- if (status != VTWERR_OK)
- bye("Can not read value at %s", m_path.path);
- ok = commit_value(act_defp, cp, creating?create_mode:update_mode, in_txn);
- if (cp)
- my_free(cp);
- goto restore;
- }
- /* else not a value */
- /* regular */
- /* do not do anything for tag type multinode */
- if (!multi_tag && creating) {
- set_at_string(child); /* for expand inside actions */
- if (do_begin) {
- status = execute_list(act_defp->
- actions[begin_act].
- vtw_list_head, act_defp);
- if (!status) {
-#ifdef DEBUG
- bye("begin action not NULL for %s\n", get_at_string());
-#else
- return(FALSE);
-#endif
- }
- }
-
- if (act_defp) {
- if (act_defp->actions[create_act].vtw_list_head) {
- status
- = execute_list(act_defp->actions[create_act].vtw_list_head,
- act_defp);
- if (!status) {
- return (FALSE);
- }
- } else if (act_defp->actions[update_act].vtw_list_head) {
- /* no create action => use update action */
- status
- = execute_list(act_defp->actions[update_act].vtw_list_head,
- act_defp);
- if (!status) {
- return (FALSE);
- }
- }
- /* not trying activate action here (activate after children are
- * configured)
- */
- }
- }
- if (creating && !in_txn) /* ELSE WAIT TILL THE END OF TXN */
- perform_create_node();
- /* children */
- ok = commit_update_children(my_defp, creating, in_txn, &update_pending);
- if (!ok)
- return(FALSE);
-
- if (update_pending){
- if (!multi_tag && act_defp &&
- act_defp->actions[update_act].vtw_list_head){
- set_at_string(child); /* for expand inside actions */
- ok = execute_list(act_defp->actions[update_act].
- vtw_list_head, act_defp);
- if (!ok)
- return(FALSE);
- /* update_pending = FALSE; */
- } else
- *update_parent = TRUE;
- }
- if (creating && !multi_tag && act_defp &&
- act_defp->actions[activate_act].vtw_list_head){
- set_at_string(child); /* for expand inside actions */
- ok = execute_list(act_defp->actions[activate_act].
- vtw_list_head, act_defp);
- /* ignore result */
- }
- if (!in_txn) /* ELSE WAIT TILL THE END OF TXN */
- perform_delete_node();
- restore:
- if (do_end){
- set_at_string(child);
- ok = execute_list(act_defp->actions[end_act].
- vtw_list_head, act_defp);
- }
-#if BITWISE
- if (do_txn && ok) {
- int len;
- char *command;
- char format1[]="rm -r -f %s/*;cp -r -f %s/%s %s";
- char format2[]="rm -r -f %s/%s;mv %s/%s %s";
- char format3[]="rm -r -f %s/%s";
- restore_paths(&mark);
- switch_path(MPATH);
- len = sizeof(format1) + 2 * strlen(get_tmpp()) +
- strlen(m_path.path) + strlen(child);
- command = malloc(len);
- sprintf(command, format1, get_tmpp(), m_path.path, child,
- get_tmpp());
- system(command);
-
- switch_path(APATH);
- len = sizeof(format2) + 2 *strlen(m_path.path) +
- 2 * strlen( child) + strlen(get_tmpp());
- command = realloc(command, len);
- sprintf(command, format2, m_path.path, child, get_tmpp(),
- child, m_path.path);
- system(command);
-
- switch_path(CPATH);
- len = sizeof(format3) + strlen(m_path.path) +
- strlen( child);
- command = realloc(command, len);
- sprintf(command, format3, m_path.path, child);
- system(command);
- free(command);
- }
-#endif
- restore_paths(&mark);
- if (my_defp && my_defp != pdefp)
- free_def(my_defp);
- return ok;
-}
-
-
-/*************************************************
- commit_delete_child:
- preconditions: t_path and c_path set up for parent
- pdefp (IN) - parent definition (may be NULL)
- child (IN) - name of the child
- do_del (IN) - if FALSE we are looking for delete target
- if TRUE, we found and switched to A (working) PATH
- commit deleted child and its descendants
- returns TRUE if no errors
- exit if error during execution
-*/
-static boolean commit_delete_child(vtw_def *pdefp, char *child,
- boolean deleting, boolean in_txn)
-{
- boolean multi_tag = FALSE;
- struct stat statbuf;
- int status;
- vtw_def def;
- vtw_def *my_defp; /*def to be given to my children */
- vtw_def *act_defp;/*def to be used for actions */
- char *cp;
- vtw_mark mark;
- boolean ok, do_txn = FALSE;
- int st;
- ok = TRUE;
-
-#ifdef DEBUG
- printf("commiting directory (node_cnt %d, free_node_cnt %d)\n"
- "mpath |%s|\n",
- node_cnt, free_node_cnt, t_path.path);
-#endif
-
- /* deleted subdirectory */
- if (strncmp(child, ".wh.", 4) == 0) { /* whiteout */
- /* ignore aufs control files */
- if (strncmp(child, ".wh..wh.", 8) == 0)
- return TRUE;
-
- if (deleting)
- /* in do_delete mode we traverse A hierarchy, no white-outs possible */
- INTERNAL; /* it is exit */
- else {
- /* deal with counterpart in working */
- switch_path(APATH);
- push_path(&m_path, child+4);
- st = lstat(m_path.path, &statbuf);
- pop_path(&m_path);
- if (st >= 0){
- /*get rid of ".wh. part in child name"*/
- /* deleting mode will handle txn, both
- begin and end
- */
- ok = commit_delete_child(pdefp, child + 4, TRUE, in_txn);
- } else {
- /* whiteout entry exists but can't see it race or unionfs problem? */
- printf("commit delete confused by lstat(%s) %s\n",
- m_path.path, strerror(errno));
- ok = TRUE;
- }
- switch_path(CPATH);
- if (ok) {
- /* delete whiteout */
- if (!in_txn){ /*ELSE WAIT TILL THE END OF TXN*/
- push_path(&m_path, child);
- perform_delete_node();
- pop_path(&m_path);
- }
- }
- return ok;
- }
- /* done with whiteouts */
- }
- /* not white out */
- mark_paths(&mark);
- if (!deleting) {
- int status;
- push_path(&m_path, child);
- switch_path(APATH); /* switch to active */
- status = lstat(m_path.path, &statbuf);
- switch_path(CPATH); /* back to changes */
- pop_path(&m_path);
- if (status < 0) {
- /* node doesn't exist in active config. it's newly created
- * so we don't need to handle delete. update will handle the
- * transaction (if any).
- */
- return TRUE;
- }
- }
- /* find our definition */
- if (pdefp && pdefp->tag) {
- /* parent is a tag, node is a tag value node */
- my_defp = NULL;
- act_defp = pdefp;
- push_path(&t_path,tag_name);
- } else {
- push_path(&t_path, child);
- push_path(&t_path, def_name);
- if ((lstat(t_path.path, &statbuf) >= 0) &&
- ((statbuf.st_mode & S_IFMT) == S_IFREG)) {
- /* defniition present */
- act_defp = my_defp = &def;
- memset(&def, 0, sizeof(def));
-#ifdef DEBUG1
- printf("Parsing definition\n");
-#endif
- if ((status = parse_def(&def, t_path.path, FALSE))) {
- bye("parse error: %d\n", status);
- }
- my_defp = act_defp = &def;
- if (def.tag)
- multi_tag = TRUE; /* tag node itself*/
- } else {
- my_defp = act_defp = NULL;
- }
- pop_path(&t_path); /*def_name */
- }
- push_path(&m_path, child);
- if (!multi_tag) {
- set_at_string(child); /* for expand inside actions */
- /* deal with txn */
- if (!in_txn && act_defp &&
- (act_defp->actions[begin_act].vtw_list_head ||
- act_defp->actions[end_act].vtw_list_head)) {
- /* if we are here we have change,
- we either in do_del and our node is change node
- or we are in change directory and our node is
- change node also */
- if (deleting)
- /* if not deleting, update will handle values */
- do_txn = TRUE;
- in_txn = TRUE;
- if (act_defp->actions[begin_act].vtw_list_head){
- status = execute_list(act_defp->actions[begin_act].
- vtw_list_head, act_defp);
- if (!status) {
-#ifdef DEBUG
- bye("begin action not NULL for %s\n", get_at_string());
-#else
- return(FALSE);
-#endif
- }
- }
- }
- }
- /* are we a "value" parent node */
- if (act_defp && !act_defp->tag && act_defp->def_type != ERROR_TYPE ) {
- /* we are value node */
- if (!act_defp->multi) {
- /* single node */
- if (!deleting) {
- /* if it was whiteout, it was converted to do_del_mode */
- restore_paths(&mark);
- return ok;
- }
-
- /* do we have actions */
- if (act_defp->actions[delete_act].vtw_list_head){
- status = get_value_to_at_string(&m_path);
- if (status != VTWERR_OK)
- bye("Can not read value at %s", m_path.path);
- /* remove \n at the line end */
- cp = strchr(get_at_string(), '\n');
- if (cp)
- *cp = 0;
- if (act_defp->actions[delete_act].vtw_list_head) {
- set_in_delete_action(TRUE);
- status = execute_list(act_defp->actions[delete_act].
- vtw_list_head, act_defp);
- set_in_delete_action(FALSE);
- if (!status) {
-#ifdef DEBUG
- bye("begin action not NULL for %s\n", get_at_string());
-#else
- return(FALSE);
-#endif
- }
- }
- free_at_string();
- }
- /* now handle commit value */
- if (!in_txn) /* ELSE WAIT TILL THE END OF TXN */
- perform_delete_node();
- goto restore;
- }
- /* else multi_node */
- cp = NULL;
- status = get_value(&cp, &m_path);
- if (status != VTWERR_OK)
- bye("Can not read value at %s", m_path.path);
- ok = commit_value(act_defp, cp, deleting?do_del_mode:del_mode,in_txn);
- if (cp)
- my_free(cp);
- goto restore;
- }
- /* else not a value */
- /* regular */
-
- /* children */
- ok = commit_delete_children(my_defp, deleting, in_txn);
- if (!ok) {
- goto restore;
- }
-
- /* do not do anything for tag itself, all action belong to values */
- if (!multi_tag) {
- set_at_string(child); /* for expand inside actions */
- if (deleting) {
- if (act_defp &&
- act_defp->actions[delete_act].vtw_list_head){
- set_in_delete_action(TRUE);
- status = execute_list(act_defp->actions[delete_act].
- vtw_list_head, act_defp);
- set_in_delete_action(FALSE);
- if (!status) {
-#ifdef DEBUG
- bye("begin action not NULL for %s\n", get_at_string());
-#else
- return(FALSE);
-#endif
- }
- }
- }
- }
-
- if (deleting) {
- if (do_txn && act_defp &&
- act_defp->actions[end_act].vtw_list_head) {
- set_at_string(child);
- ok = execute_list(act_defp->actions[end_act].
- vtw_list_head, act_defp);
- if (!ok)
- goto restore;
- }
- /* delete node and all its descendants */
- if (!in_txn || do_txn)/* ELSE WAIT TILL THE END OF TXN */
- perform_delete_node();
- }
- restore:
- restore_paths(&mark);
- if (my_defp && my_defp != pdefp)
- free_def(my_defp);
- return ok;
-}
-
-/* add a value to the valstruct. struct must be initialized (memset 0).
- * cp: pointer to value
- * type: type of value
- */
-static void
-valstruct_append(valstruct *mvals, char *cp, vtw_type_e type)
-{
- if (!(mvals->free_me)) {
- /* empty struct. add 1st value */
- mvals->free_me = TRUE;
- mvals->val = cp;
- mvals->val_type = type;
- } else {
- if ((mvals->cnt % MULTI_ALLOC) == 0) {
- /* convert into multivalue */
- mvals->vals = my_realloc(mvals->vals, (mvals->cnt + MULTI_ALLOC)
- * sizeof(char *), "add_value");
- if (mvals->cnt == 0) { /* single value - convert */
- mvals->vals[0] = mvals->val;
- mvals->cnt= 1;
- mvals->val = NULL;
- }
- }
- mvals->vals[mvals->cnt] = cp;
- ++mvals->cnt;
- }
-}
-
-/* get the filtered directory listing and put the names in valstruct.
- * dp: target directory
- * mvals: structure for storing the names
- * type: type of the names
- * exclude_wh: exclude whiteouts
- */
-static void
-get_filtered_directory_listing(DIR *dp, valstruct *mvals, vtw_type_e type,
- int exclude_wh, int sort_order)
-{
- char *dname = NULL;
- char *cp = NULL;
-
- memset(mvals, 0, sizeof (valstruct));
- struct DirIndex *di = init_next_filtered_dirname(dp, sort_order, exclude_wh);
- while ((dname = get_next_filtered_dirname(di)) != NULL) {
- cp = clind_unescape(dname);
- valstruct_append(mvals, cp, type);
- }
- release_dir_index(di);
-}
-
-/* check if a value is one of those in a valstruct.
- * cp: pointer to value
- * vals: pointer to valstruct
- *
- * return 1 if yes, 0 otherwise.
- * TODO: optimization
- */
-static int
-is_val_in_valstruct(char *cp, valstruct *vals)
-{
- int i = 0;
- if (!(vals->free_me)) {
- /* empty struct. */
- return 0;
- }
- if (vals->cnt == 0) {
- /* single-value struct */
- if (strcmp(cp, vals->val) == 0) {
- return 1;
- }
- return 0;
- }
- /* multi-value */
- for (i = 0; i < vals->cnt; i++) {
- if (strcmp(cp, vals->vals[i]) == 0) {
- return 1;
- }
- }
- return 0;
-}
-
-/* returns the value at a particular index in a valstruct.
- * vals: pointer to the valstruct
- * idx: index of the value
- *
- * return pointer to the value, NULL if idx is not valid.
- */
-static char *
-idx_val_in_valstruct(valstruct *vals, int idx)
-{
- if (!(vals->free_me)) {
- /* empty struct. */
- return NULL;
- }
- if (vals->cnt == 0) {
- /* single-value struct */
- if (idx == 0) {
- return vals->val;
- }
- return NULL;
- }
- /* multi-value */
- if ((idx < 0) || (idx >= vals->cnt)) {
- return NULL;
- }
- return vals->vals[idx];
-}
-
-
-static boolean commit_delete_children(vtw_def *defp, boolean deleting,
- boolean in_txn)
-{
- DIR *dp, *adp = NULL, *mdp = NULL;
- boolean ok = TRUE;
- char *child;
- vtw_type_e type;
- valstruct mvals, valsA, valsM;
- int elem, curi;
- vtw_sorted cur_sorted;
-
- if (!deleting) {
- switch_path(APATH); /* switch to active */
- if ((adp = opendir(m_path.path)) == NULL) {
- INTERNAL;
- }
- switch_path(MPATH); /* switch to modified */
- if ((mdp = opendir(m_path.path)) == NULL) {
- INTERNAL;
- }
- switch_path(CPATH); /* back to changes */
- }
-
- if ((dp = opendir(m_path.path)) == NULL){
- if (deleting) {
- /* deleting. adp & mdp are not opened so no need to close. */
- return TRUE;
- }
- INTERNAL;
- }
- if (defp)
- type = defp->def_type;
- else
- type = TEXT_TYPE;
- if (type == ERROR_TYPE)
- type = TEXT_TYPE;
- memset(&mvals, 0, sizeof (valstruct));
- memset(&valsA, 0, sizeof (valstruct));
- memset(&valsM, 0, sizeof (valstruct));
- memset(&cur_sorted, 0, sizeof(vtw_sorted));
-
- /* changes directory */
- get_filtered_directory_listing(dp, &mvals, type, 0, 1);
- if (closedir(dp) != 0) {
- INTERNAL;
- }
-
- if (adp && mdp) {
- /* active directory */
- get_filtered_directory_listing(adp, &valsA, type, 0, 1);
- if (closedir(adp) != 0) {
- INTERNAL;
- }
- /* modified directory */
- get_filtered_directory_listing(mdp, &valsM, type, 0, 1);
- if (closedir(mdp) != 0) {
- INTERNAL;
- }
-
- if (valsA.free_me) {
- /* A is not empty */
- int idx = 0;
- char *cp = NULL;
- for (idx = 0; (cp = idx_val_in_valstruct(&valsA, idx)); idx++) {
- if (!is_val_in_valstruct(cp, &valsM)) {
- /* cp is in A but not in M */
- /* construct whiteout name */
- char *wh_name = my_malloc(strlen(cp) + 4 + 1, "del_children");
- strcpy(wh_name, ".wh.");
- strcpy(wh_name + 4, cp);
- if (!is_val_in_valstruct(wh_name, &mvals)) {
- /* whiteout not in C */
- /* add whiteout to mvals */
- valstruct_append(&mvals, wh_name, type);
- }
- }
- }
- free_val(&valsA);
- }
- if (valsM.free_me) {
- free_val(&valsM);
- }
- }
-
- if (!(mvals.free_me)) {
- /* empty struct. nothing to do. */
- return TRUE;
- }
- vtw_sort(&mvals, &cur_sorted);
- for (curi = 0; curi < cur_sorted.num && ok; ++curi){
- if (type == TEXT_TYPE ||
- type == BOOL_TYPE)
- child = (char *)(cur_sorted.ptrs[curi]);
- else {
- elem = (((unsigned int *)(cur_sorted.ptrs[curi]))-
- cur_sorted.parts)/
- cur_sorted.partnum;
- child = mvals.cnt?mvals.vals[elem]:
- mvals.val;
- }
- ok = commit_delete_child(defp, child, deleting, in_txn);
- }
- free_val(&mvals);
- free_sorted(&cur_sorted);
- return ok;
-}
-
-static boolean commit_update_children(vtw_def *defp, boolean creating,
- boolean in_txn, boolean *parent_update)
-{
- DIR *dp;
- boolean ok = TRUE;
- char *child;
- vtw_type_e type;
- valstruct mvals;
- int elem, curi;
- vtw_sorted cur_sorted;
-
- if ((dp = opendir(m_path.path)) == NULL){
- printf("%s:%d: opendir error: path=%s\n",
- __FUNCTION__,__LINE__,m_path.path);
- INTERNAL;
- }
-
- memset(&mvals, 0, sizeof (valstruct));
- memset(&cur_sorted, 0, sizeof(vtw_sorted));
- if (defp)
- type = defp->def_type;
- else
- type = TEXT_TYPE;
- if (type == ERROR_TYPE)
- type = TEXT_TYPE;
-
- get_filtered_directory_listing(dp, &mvals, type, 0, 0);
- if (closedir(dp) != 0) {
- INTERNAL;
- }
-
- if (!(mvals.free_me)) {
- /* empty struct. nothing to do. */
- return TRUE;
- }
- vtw_sort(&mvals, &cur_sorted);
- for (curi = 0; curi < cur_sorted.num && ok; ++curi){
- if (type == TEXT_TYPE ||
- type == BOOL_TYPE)
- child = (char *)(cur_sorted.ptrs[curi]);
- else {
- elem = (((unsigned int *)(cur_sorted.ptrs[curi]))-
- cur_sorted.parts)/
- cur_sorted.partnum;
- child = mvals.cnt?mvals.vals[elem]:
- mvals.val;
- }
- ok = commit_update_child(defp, child, creating, in_txn, parent_update);
- }
- free_val(&mvals);
- free_sorted(&cur_sorted);
- return ok;
-}
-
-
-/*************************************************
- commit_value:
- executes commit for the value leave node
-**************************************************/
-static boolean commit_value(vtw_def *defp, char *cp,
- vtw_cmode mode, boolean in_txn)
-{
-
- valstruct act_value;
- int status;
- int curi,acti, partnum, res=0;
- void *actp, *curp;
- boolean no_shadow;
- boolean ok;
- int total, a_res, c_res;
- char **a_ptr, **c_ptr, *val_string;
- boolean cur_pr_val;
- int pr_index;
- int sign;
- boolean creating;
- vtw_node *actions;
- valstruct cur_value;
- vtw_sorted cur_sorted;
- vtw_sorted act_sorted;
-
- ok = TRUE;
- actions = NULL;
- if(mode == del_mode || mode == do_del_mode) {
- creating = FALSE;
- if (defp && defp->actions[delete_act].vtw_list_head) {
- set_in_delete_action(TRUE);
- actions = defp->actions[delete_act].vtw_list_head;
- }
- } else {
- creating = TRUE;
- if (defp && defp->actions[create_act].vtw_list_head)
- actions = defp->actions[create_act].vtw_list_head;
- }
- /* prepare cur_value */
-
- status = char2val(defp, cp, &cur_value);
- if (mode != do_del_mode && mode != create_mode) {
- /* get active value */
- switch_path(APATH); /* switch form CCD to ACD */
- status = get_value(&cp, &m_path);
- switch_path(CPATH); /* back to CCD */
- if (status != VTWERR_OK) {
- no_shadow = TRUE;
- }else
- no_shadow = FALSE;
- } else {
- no_shadow = TRUE;
- }
- vtw_sort(&cur_value, &cur_sorted);
- if(no_shadow) {
- act_sorted.num = 0;
- }else {
- status = char2val(defp, cp, &act_value);
- if (status != VTWERR_OK) {
- INTERNAL;
- }
- /* sort them */
- vtw_sort(&act_value, &act_sorted);
- }
- if (mode == do_del_mode) {
- /* it was actually act_sorted, not cur_sorted */
- act_sorted = cur_sorted;
- cur_sorted.num = 0;
- act_value = cur_value;
- /* act_value will be freed by freeing cur_value
- do not zero out it here */
- }
- acti = 0;
- curi = 0;
- total = act_sorted.num + cur_sorted.num;
- a_res=0;
- c_res=0;
- a_ptr = my_malloc(total*sizeof(char *), "");
- c_ptr = my_malloc(total*sizeof(char *), "");
- while (acti < act_sorted.num || curi < cur_sorted.num) {
- if (acti == act_sorted.num) {
- cur_pr_val = TRUE;
- pr_index = curi;
- sign = +1;
- ++curi;
- } else if (curi == cur_sorted.num) {
- cur_pr_val = FALSE;
- pr_index = acti;
- sign = -1;
- ++acti;
- } else {
- /* compare */
- actp = act_sorted.ptrs[acti];
- curp = cur_sorted.ptrs[curi];
- /* compare */
- if (act_sorted.partnum){
- for(partnum = 0; partnum < act_sorted.partnum;
- ++partnum) {
- res = *((int *)actp + partnum) -
- *((int *)curp + partnum);
- if (res)
- break;
- }
- } else{
- res = strcmp((char *)actp, (char *) curp);
- }
- if (res == 0) {
- /* the same */
- cur_pr_val = TRUE;
- pr_index = curi;
- sign = 0;
- ++acti;
- ++curi;
- } else if (res < 0) {
- /* act < cur, act is unmatched */
- cur_pr_val = FALSE;
- pr_index = acti;
- sign = -1;
- ++acti;
- }else {
- /* cur < act, cur is unmatched */
- cur_pr_val = TRUE;
- pr_index = curi;
- sign = 1;
- ++curi;
- }
- }
- if (defp->def_type == TEXT_TYPE ||
- defp->def_type == BOOL_TYPE) {
- val_string = cur_pr_val?
- ((char *)(cur_sorted.ptrs[pr_index])):
- ((char *)(act_sorted.ptrs[pr_index]));
- } else {
- if (cur_pr_val) {
- int elem = (((unsigned int *)(cur_sorted.ptrs[pr_index]))-
- cur_sorted.parts)/
- cur_sorted.partnum;
- val_string = cur_value.cnt?cur_value.vals[elem]:
- cur_value.val;
- } else {
- int elem = (((unsigned int *)(act_sorted.ptrs[pr_index]))-
- act_sorted.parts)/
- act_sorted.partnum;
- val_string = act_value.cnt?act_value.vals[elem]:
- act_value.val;
- }
- }
- set_at_string(val_string);
- switch (sign) {
- case 0: /* found in both, no actions, include in both */
- a_ptr[a_res++]=val_string;
- c_ptr[c_res++]=val_string;
- break;
- case 1: /* found only in change */
- if (ok && creating) {
- if (actions) {
- /* do create action */
- ok = execute_list(actions, defp);
- } else if (defp && defp->actions[update_act].vtw_list_head) {
- /* no create action => use update action */
- ok = execute_list(defp->actions[update_act].vtw_list_head, defp);
- }
- if (ok && defp && defp->actions[activate_act].vtw_list_head) {
- /* try activate action */
- ok = execute_list(defp->actions[activate_act].vtw_list_head, defp);
- }
- /* if succ, make it look old */
- if(ok)
- a_ptr[a_res++]=val_string;
- }
- c_ptr[c_res++]=val_string; /* in all cases */
- break;
- case -1: /* found only in working */
- if (ok && !creating && actions) {/* ok and deleting */
- ok = execute_list(actions, defp);
- }
- /* if succ and deleting - do nothing, else */
- if (!ok || creating)
- a_ptr[a_res++]=val_string;
- }
- }
- if (creating && ok)
- c_res = 0;
-#if BITWISE
- if (!in_txn) {/* ELSE WAIT TILL THE END OF TXN */
- switch_path(APATH);
- if (a_res) {
- make_dir();
- push_path(&m_path, VAL_NAME);
- fp = fopen(m_path.path, "w");
- if (fp == NULL)
- bye("Can not open value file %s", m_path.path);
- for(i=0;i<a_res;++i)
- if (fputs(a_ptr[i], fp) < 0 || fputc('\n',fp) < 0)
- bye("Error writing file %s", m_path.path);
- fclose(fp);
- pop_path(&m_path);
- }else{
- perform_delete_node();
- }
- switch_path(CPATH);
- if (c_res) {
- make_dir();
- push_path(&m_path, VAL_NAME);
- fp = fopen(m_path.path, "w");
- if (fp == NULL)
- bye("Can not open value file %s", m_path.path);
- for(i=0;i<c_res;++i)
- if (fputs(c_ptr[i], fp) < 0 || fputc('\n',fp) < 0)
- bye("Error writing file %s", m_path.path);
- fclose(fp);
- pop_path(&m_path);
- }else{
- perform_delete_node();
- }
- }
-#endif /*BITWISE*/
- if (mode == do_del_mode)
- switch_path(APATH);
- else
- switch_path(CPATH);
- if(act_sorted.num)
- free_sorted(&act_sorted);
- if(cur_sorted.num)
- free_sorted(&cur_sorted);
- my_free(a_ptr);
- my_free(c_ptr);
- free_val(&cur_value);
- if (!no_shadow)
- free_val(&act_value);
- set_in_delete_action(FALSE);
- return ok;
-}
-
-static int fin_commit(boolean ok)
-{
- char *command;
- static const char format1[]="cp -r -f %s/* %s"; /*mdirp, tmpp*/
- static const char format2[]="sudo umount %s"; /*mdirp*/
- static const char format3[]="rm -f %s/" MOD_NAME " >&/dev/null ; /bin/true";
- /*tmpp*/
- static const char format4[]="rm -rf %s/{.*,*} >&/dev/null ; /bin/true"; /*cdirp*/
- static const char format5[]="rm -rf %s/{.*,*} >&/dev/null ; /bin/true"; /*adirp*/
- static const char format6[]="mv -f %s/* -t %s";/*tmpp, adirp*/
- static const char format7[]="sudo mount -t $UNIONFS -o dirs=%s=rw:%s=ro"
- " $UNIONFS %s"; /*cdirp, adirp, mdirp*/
- int m_len = strlen(get_mdirp());
- int t_len = strlen(get_tmpp());
- int c_len = strlen(get_cdirp());
- int a_len = strlen(get_adirp());
- set_echo(TRUE);
- if (!ok){
- fprintf(out_stream, "Commit failed\n");
- return -1;
- }
- command = malloc(strlen(format1) + m_len + t_len);
- sprintf(command, format1, get_mdirp(), get_tmpp());
- system(command);
-
- command = realloc(command, strlen(format2) + m_len);
- sprintf(command, format2, get_mdirp());
- system(command);
-
- command = realloc(command, strlen(format3) + t_len);
- sprintf(command, format3, get_tmpp());
- system(command);
-
- command = realloc(command, strlen(format4) + c_len);
- sprintf(command, format4, get_cdirp());
- system(command);
-
- command = realloc(command, strlen(format5) + a_len);
- sprintf(command, format5, get_adirp());
- system(command);
-
- command = realloc(command, strlen(format6) + t_len + a_len);
- sprintf(command, format6, get_tmpp(), get_adirp());
- system(command);
-
- command = realloc(command, strlen(format7) + c_len + a_len + m_len);
- sprintf(command, format7, get_cdirp(), get_adirp(), get_mdirp());
- system(command);
- free(command);
-
- /* notify other users in config mode */
- system("/opt/vyatta/sbin/vyatta-cfg-notify");
-
- return 0;
-}
-