diff options
author | An-Cheng Huang <ancheng@vyatta.com> | 2008-04-03 15:09:34 -0700 |
---|---|---|
committer | An-Cheng Huang <ancheng@vyatta.com> | 2008-04-03 15:09:34 -0700 |
commit | 9158b59fd7cdd018a5c9b1e36c097b0436fd02a4 (patch) | |
tree | bc6a8afbe6e51d9f98825d2d02b677ad733c5c16 | |
parent | bcdf7826a9e29cd48627ffb57b665fcad9a714fa (diff) | |
download | vyatta-cfg-9158b59fd7cdd018a5c9b1e36c097b0436fd02a4.tar.gz vyatta-cfg-9158b59fd7cdd018a5c9b1e36c097b0436fd02a4.zip |
fix for bug 3097: correctly detect deleted tag
-rw-r--r-- | src/commit.c | 96 |
1 files changed, 88 insertions, 8 deletions
diff --git a/src/commit.c b/src/commit.c index 6ed8632..534757c 100644 --- a/src/commit.c +++ b/src/commit.c @@ -972,6 +972,65 @@ get_filtered_directory_listing(DIR *dp, valstruct *mvals, vtw_type_e type, } } +/* 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) { @@ -1013,32 +1072,53 @@ static boolean commit_delete_children(vtw_def *defp, boolean deleting, memset(&valsM, 0, sizeof (valstruct)); memset(&cur_sorted, 0, sizeof(vtw_sorted)); + /* changes directory */ get_filtered_directory_listing(dp, &mvals, type, 0); if (closedir(dp) != 0) { INTERNAL; } - if (adp) { + + if (adp && mdp) { + /* active directory */ get_filtered_directory_listing(adp, &valsA, type, 0); if (closedir(adp) != 0) { INTERNAL; } - } - if (mdp) { + /* modified directory */ get_filtered_directory_listing(mdp, &valsM, type, 0); 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; } - /* XXX TODO */ - /* for each "node" in A but not in M - * if ".wh.node" is not in mvals - * add ".wh.node" to mvals - */ vtw_sort(&mvals, &cur_sorted); for (curi = 0; curi < cur_sorted.num && ok; ++curi){ if (type == TEXT_TYPE || |