From 2643994886fcb691f0def1eff5a91c693eb3fa82 Mon Sep 17 00:00:00 2001 From: Lon Hohberger Date: Fri, 8 Jan 2010 16:40:31 -0500 Subject: [PATCH] Better config query & multiple value/tag support Signed-off-by: Lon Hohberger --- config/config-stack.h | 5 +- config/config.y | 33 +++++++++++-- config/simpleconfig.c | 112 ++++++++++++++++++++++++++++++++++++------ 3 files changed, 129 insertions(+), 21 deletions(-) diff --git a/config/config-stack.h b/config/config-stack.h index ebc7f7c..1eb3cfa 100644 --- a/config/config-stack.h +++ b/config/config-stack.h @@ -13,6 +13,7 @@ struct value { struct node { char *id; + char *val; struct node *nodes; struct value *values; struct node *next; @@ -30,8 +31,8 @@ extern struct node *node_list; extern struct parser_context *context_stack; int _sc_value_add(char *id, char *val, struct value **list); -int _sc_node_add(char *id, struct value *vallist, struct node *nodelist, - struct node **list); +int _sc_node_add(char *id, char *val, struct value *vallist, + struct node *nodelist, struct node **list); #endif diff --git a/config/config.y b/config/config.y index e2013e8..c80c0c2 100644 --- a/config/config.y +++ b/config/config.y @@ -32,8 +32,8 @@ _sc_value_add(char *id, char *val, struct value **list) int -_sc_node_add(char *id, struct value *vallist, struct node *nodelist, - struct node **list) +_sc_node_add(char *id, char *val, struct value *vallist, + struct node *nodelist, struct node **list) { struct node *n; @@ -45,6 +45,7 @@ _sc_node_add(char *id, struct value *vallist, struct node *nodelist, memset(n, 0, sizeof(*n)); //snprintf(n->id, sizeof(n->id), "%s", id); n->id = id; /* malloc'd during parsing */ + n->val = val; /* malloc'd during parsing */ n->values = vallist; n->nodes = nodelist; n->next = *list; @@ -72,7 +73,19 @@ node: struct parser_context *c = NULL; c = context_stack; - _sc_node_add($1, val_list, node_list, &c->node_list); + _sc_node_add($1, NULL, val_list, node_list, &c->node_list); + val_list = c->val_list; + node_list = c->node_list; + context_stack = c->next; + + free(c); + } + | + T_ID T_EQ T_VAL T_OBRACE stuff T_CBRACE { + struct parser_context *c = NULL; + + c = context_stack; + _sc_node_add($1, $3, val_list, node_list, &c->node_list); val_list = c->val_list; node_list = c->node_list; context_stack = c->next; @@ -84,7 +97,19 @@ node: struct parser_context *c = NULL; c = context_stack; - _sc_node_add($1, val_list, node_list, &c->node_list); + _sc_node_add($1, NULL, val_list, node_list, &c->node_list); + val_list = c->val_list; + node_list = c->node_list; + context_stack = c->next; + + free(c); + } + | + T_ID T_EQ T_VAL T_OBRACE T_CBRACE { + struct parser_context *c = NULL; + + c = context_stack; + _sc_node_add($1, $3, val_list, node_list, &c->node_list); val_list = c->val_list; node_list = c->node_list; context_stack = c->next; diff --git a/config/simpleconfig.c b/config/simpleconfig.c index 8de1184..f5037a5 100644 --- a/config/simpleconfig.c +++ b/config/simpleconfig.c @@ -2,6 +2,8 @@ #include #include #include +#include +#include #include "simpleconfig.h" #include "config-stack.h" @@ -37,7 +39,12 @@ _sc_dump_d(struct node *node, int depth, FILE *fp) for (x = 0; x < depth; x++) fprintf(fp, "\t"); - fprintf(fp, "%s {\n", node->id); + + if (node->val) { + fprintf(fp, "%s = \"%s\" {\n", node->id, node->val); + } else { + fprintf(fp, "%s {\n", node->id); + } for (n = node->nodes; n; n = n->next) { _sc_dump_d(n, depth+1, fp); @@ -161,45 +168,120 @@ _sc_get(config_info_t *config, const char *key, char *value, size_t valuesz) struct value *v, *values = ((struct parser_context *)config)->val_list; char *ptr; char *slash; + char *bracket; + char *id; + int req_index = 0; + int curr_index = 0; int found; if (!config) - return 1; + return -1; + assert(strlen(key) < sizeof(buf)); ptr = (char *)key; +top: while ((slash = strchr(ptr, '/'))) { memset(buf, 0, sizeof(buf)); strncpy(buf, ptr, (slash - ptr)); ptr = ++slash; + id = NULL; + bracket = strchr(buf, '['); + if (bracket) { + *bracket = 0; + ++bracket; + + id = bracket; + + bracket = strchr(bracket, ']'); + if (!bracket) + return 1; + *bracket = 0; + + if (id[0] == '@') { + ++id; + if (!strlen(id)) { + return 1; + } + } else { + req_index = atoi(id); + if (req_index <= 0) + return 1; + id = NULL; + } + } + found = 0; + curr_index = 0; for (n = node; n; n = n->next) { - if (!strcasecmp(n->id, buf)) { - node = n->nodes; - values = n->values; - found = 1; - break; + + if (strcasecmp(n->id, buf)) + continue; + + ++curr_index; + + if (req_index && (curr_index != req_index)) { + continue; + } else if (id && strcasecmp(n->val, id)) { + continue; } + + node = n->nodes; + values = n->values; + found = 1; + break; } if (!found) return 1; } - if (ptr[0] != '@') + if (ptr[0] != '@') { + if (node->val) { + strncpy(value, node->val, valuesz); + return 0; + } return 1; + } ++ptr; found = 0; + id = NULL; + + strncpy(buf, ptr, sizeof(buf)); + bracket = strchr(buf, '['); + + req_index = 0; + curr_index = 0; + + if (bracket) { + *bracket = 0; + ++bracket; + + id = bracket; + + bracket = strchr(bracket, ']'); + if (!bracket) + return 1; + *bracket = 0; + + req_index = atoi(id); + if (req_index <= 0) + return 1; + id = NULL; + } for (v = values; v; v = v->next) { - if (!strcasecmp(v->id, ptr)) { - if (v->val == NULL) - return 1; - snprintf(value, valuesz, "%s", v->val); - return 0; - } + + if (strcasecmp(v->id, buf)) + continue; + + ++curr_index; + if (req_index && (curr_index != req_index)) + continue; + snprintf(value, valuesz, "%s", v->val); + return 0; } return 1; @@ -238,7 +320,7 @@ _sc_set(config_info_t *config, const char *key, const char *value) id_dup = strdup(buf); if (!id_dup) return -1; - _sc_node_add(id_dup, NULL, NULL, nodes); + _sc_node_add(id_dup, NULL, NULL, NULL, nodes); n = *nodes; nodes = &n->nodes; values = &n->values;