summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--src/commit2.c9
-rw-r--r--src/common/common.h2
-rw-r--r--src/common/unionfs.c112
3 files changed, 100 insertions, 23 deletions
diff --git a/src/commit2.c b/src/commit2.c
index 0a332d2..bdac39e 100644
--- a/src/commit2.c
+++ b/src/commit2.c
@@ -156,15 +156,12 @@ main(int argc, char** argv)
//get local session data plus configuration data
GNode *config_data = common_get_local_session_data();
if (g_node_n_children(config_data) == 0) {
- common_commit_clean_temp_config(test_mode);
+ common_commit_clean_temp_config(NULL, test_mode);
fprintf(out_stream, "No configuration changes to commit\n");
return 0;
}
- GNode *orig_node_tree = NULL;
- if (disable_partial_commit == TRUE) {
- orig_node_tree = g_node_copy(config_data);
- }
+ GNode *orig_node_tree = g_node_copy(config_data);
// Get collection of transactions, i.e. trans nodes that have been activated.
GNode *trans_coll = get_transactions(config_data, priority_mode);
@@ -251,7 +248,7 @@ main(int argc, char** argv)
* Need to add to the following func below to clean up dangling .wh. files
*/
if (g_dump_actions == FALSE) {
- common_commit_clean_temp_config(test_mode);
+ common_commit_clean_temp_config(orig_node_tree, test_mode);
}
if (g_debug == TRUE) {
printf("commit2: successful commit, now cleaning up temp directories\n");
diff --git a/src/common/common.h b/src/common/common.h
index e3b7bdc..06e3a9e 100644
--- a/src/common/common.h
+++ b/src/common/common.h
@@ -48,6 +48,6 @@ common_commit_copy_to_live_config(GNode *root_node, boolean test_mode);
*
**/
void
-common_commit_clean_temp_config(boolean test_mode);
+common_commit_clean_temp_config(GNode *root_node, boolean test_mode);
#endif //__COMMON_H__
diff --git a/src/common/unionfs.c b/src/common/unionfs.c
index 840dc68..8d02bbe 100644
--- a/src/common/unionfs.c
+++ b/src/common/unionfs.c
@@ -43,6 +43,9 @@ copy_func(GNode *node, gpointer data);
static gboolean
delete_func(GNode *node, gpointer data);
+static gboolean
+delete_wh_func(GNode *node, gpointer data);
+
static void
piecewise_copy(GNode *root_node, boolean test_mode);
@@ -613,11 +616,19 @@ common_commit_copy_to_live_config(GNode *node, boolean test_mode)
return;
}
+
+//needed for iteration below
+struct SrcDst {
+ char *_src;
+ char *_dst;
+ boolean _test_mode;
+};
+
/**
*
**/
void
-common_commit_clean_temp_config(boolean test_mode)
+common_commit_clean_temp_config(GNode *root_node, boolean test_mode)
{
if (g_debug) {
printf("common_commit_clean_temp_config()\n");
@@ -625,13 +636,6 @@ common_commit_clean_temp_config(boolean test_mode)
//first clean up the root
// common_commit_copy_to_live_config("/");
- /*
- * Need to add to the following func below to clean up dangling .wh. files.
- * This pass needs to be prior to the commands below (but after the umount).
- * This fixes a bug when higher priority root nodes are deleted and not removed.
- */
-
-
char *command;
command = malloc(MAX_LENGTH_DIR_PATH);
static const char format2[]="sudo umount %s"; //mdirp
@@ -661,6 +665,27 @@ common_commit_clean_temp_config(boolean test_mode)
system(command);
}
+ /*
+ * Need to add to the following func below to clean up dangling .wh. files.
+ * This pass needs to be prior to the commands below (but after the umount).
+ * This fixes a bug when higher priority root nodes are deleted and not removed.
+ */
+
+ //Iterate through node hierarchy and remove deleted nodes from active config--insurance
+ //to protect against priority whiteouts in parent/child order
+ //TOP DOWN
+ if (root_node != NULL) {
+ struct SrcDst sd;
+ sd._test_mode = test_mode;
+
+ g_node_traverse(root_node,
+ G_PRE_ORDER,
+ G_TRAVERSE_ALL,
+ -1,
+ (GNodeTraverseFunc)delete_wh_func,
+ (gpointer)&sd);
+ }
+
sprintf(command, format3, tbuf);
if (g_debug) {
printf("%s\n",command);
@@ -976,13 +1001,6 @@ insert_sibling_in_order(GNode *parent, GNode *child)
return new_node;
}
-//needed for iteration below
-struct SrcDst {
- char *_src;
- char *_dst;
- boolean _test_mode;
-};
-
/**
*
**/
@@ -1088,9 +1106,10 @@ delete_func(GNode *node, gpointer data)
return FALSE;
}
+ struct SrcDst *sd = (struct SrcDst*)data;
+
char *command = malloc(MAX_LENGTH_DIR_PATH);
- struct SrcDst *sd = (struct SrcDst*)data;
//DONT HAVE THE COMMAND BELOW BLOW AWAY WHITEOUT FILES!!!!!
static const char format[]="rm -f %s%s{*,.*} >&/dev/null;rmdir %s%s >&/dev/null ; /bin/true"; //need to remove opaque file.
static const char format_force_delete[]="rm -f %s%s{*,.*} >&/dev/null;rmdir %s%s >&/dev/null ; /bin/true"; //force delete as this is a delete operation with dependency
@@ -1153,3 +1172,64 @@ delete_func(GNode *node, gpointer data)
return FALSE;
}
+
+
+/**
+ *
+ *
+ **/
+static gboolean
+delete_wh_func(GNode *node, gpointer data)
+{
+ if (node == NULL) {
+ return FALSE;
+ }
+
+ char abuf[MAX_LENGTH_DIR_PATH];
+ static const char format0[]="rm -fr %s >&/dev/null ; /bin/true";
+ struct SrcDst *sd = (struct SrcDst*)data;
+
+ GNode *parent_node = node->parent;
+
+ //on node where operation is delete and parent is noop or active then remove directory from active config
+ //if this is a deletion operation, need to remove
+ if (parent_node != NULL) {
+ if (IS_DELETE(((struct VyattaNode*)(node->data))->_data._operation) &&
+ !IS_ACTIVE(((struct VyattaNode*)(node->data))->_data._operation) &&
+ !IS_DELETE(((struct VyattaNode*)(parent_node->data))->_data._operation)) {
+
+ char *path = ((struct VyattaNode*)(node->data))->_data._path;
+ sprintf(abuf,"%s%s",get_adirp(),path);
+ //mkdir temp merge
+ char command[MAX_LENGTH_DIR_PATH];
+ sprintf(command,format0,abuf);
+ if (g_debug) {
+ printf("%s\n",command);
+ fflush(NULL);
+ }
+ if (sd->_test_mode == FALSE) {
+ system(command);
+ }
+
+ }
+ }
+ else {
+ if (IS_DELETE(((struct VyattaNode*)(node->data))->_data._operation) &&
+ !IS_ACTIVE(((struct VyattaNode*)(node->data))->_data._operation)) {
+ char *path = ((struct VyattaNode*)(node->data))->_data._path;
+ sprintf(abuf,"%s%s",get_adirp(),path);
+ //mkdir temp merge
+ char command[MAX_LENGTH_DIR_PATH];
+ sprintf(command,format0,abuf);
+ if (g_debug) {
+ printf("%s\n",command);
+ fflush(NULL);
+ }
+ if (sd->_test_mode == FALSE) {
+ system(command);
+ }
+ }
+ }
+ return FALSE;
+}
+