58 (?<group> \\[ (?:[ \t]*) (?<gname>[\\w\\d_]+) (?:[ \t]*) \\]) \n|\ 59 (?<key> ([\\.\\w\\d_-]+)) (?:[ \t]*) = (?:[ \t]*)(?<value>.*)\n|\ 60 (?<klist> ([\\.\\w\\d_-]+))\\[\\](?:[ \t]*) = (?:[ \t]*)(?<vlist>.*)\n|\ 61 (?<error> (?:\\S+)(?:[ \t]*))\n";
81 static gint
str_comp(gconstpointer a, gconstpointer b, gpointer user_data)
83 return strcmp((
char*) a, (
char*) b);
97 static gboolean
collect_keys(
char* key, gpointer* value,
char** data)
113 #define BUFFER_SIZE 4096 114 #define yynext() (c = next()) != EOF 116 #define throw_error(error, domain, code, ...) \ 117 { g_set_error(error, domain, code, __VA_ARGS__); \ 132 GTree* group = (GTree*) data;
133 gchar* key = g_match_info_fetch(match, 1);
134 gchar* sub = g_tree_lookup(group, key);
136 g_string_append(ret, sub);
160 gchar* fname, guint line, GError** error)
170 "%s[line %d]: key \"%s\" does not have an associated group",
175 if ((val = g_tree_lookup(group, key)))
177 val = g_strdup_printf(
"%s[%s]", val, tmp);
182 val = g_strdup_printf(
"[%s]", tmp);
191 g_tree_insert(group, g_strdup(key), val);
215 fo_conf* dest, gchar* yyfile, guint yyline, GError** error)
217 gchar* error_t = NULL;
220 if ((error_t = g_match_info_fetch_named(match,
"error")) != NULL)
223 "%s[line %d]: incorrectly formated line \"%s\".",
224 yyfile, yyline, error_t);
229 gchar* group = g_match_info_fetch_named(match,
"group");
230 gchar* gname = g_match_info_fetch_named(match,
"gname");
231 gchar* key = g_match_info_fetch_named(match,
"key");
232 gchar* value = g_match_info_fetch_named(match,
"value");
233 gchar* klist = g_match_info_fetch_named(match,
"klist");
234 gchar* vlist = g_match_info_fetch_named(match,
"vlist");
235 gchar* wrong = g_match_info_fetch_named(match,
"error");
237 if (group != NULL && group[0])
239 *g_current = g_tree_new_full(
str_comp, NULL, g_free, g_free);
240 g_tree_insert(dest->
group_map, g_strdup(gname), *g_current);
242 else if (key != NULL && key[0])
244 if (!
fo_config_key(*g_current, key, value, FALSE, yyfile, yyline, error))
247 else if (klist != NULL && klist[0])
249 if (!
fo_config_key(*g_current, klist, vlist, TRUE, yyfile, yyline, error))
293 G_REGEX_EXTENDED | G_REGEX_OPTIMIZE, 0, NULL);
296 G_REGEX_EXTENDED | G_REGEX_OPTIMIZE, 0, NULL);
298 if ((fd = fopen(rawname,
"r")) == NULL)
300 "unable to open configuration file \"%s\"", rawname);
308 (GDestroyNotify) g_tree_unref);
310 GTree* g_current = NULL;
312 while (fgets(text,
sizeof(text), fd) != NULL)
322 g_match_info_free(match);
352 "ERROR: invalid fo_conf object passed to fo_config_get");
354 if ((tree = g_tree_lookup(conf->
group_map, group)) == NULL)
359 "ERROR: unknown group \"%s\"", group);
361 if ((ret = g_tree_lookup(tree, key)) == NULL)
366 "ERROR: unknown key=\"%s\" for group=\"%s\"", key, group);
368 return g_tree_lookup(tree, key);
400 "ERROR: invalid fo_conf object passed to fo_config_get_list");
407 "ERROR: %s[%s] must be of type list to get list element", group, key);
414 "ERROR: %s[%s] %d is out of range", group, key, idx);
420 g_tree_lookup(conf->
group_map, group), key);
423 for (depth = 0; depth < idx;)
425 while (*(++curr) !=
'[');
430 while (*(++curr) !=
']');
431 val = g_strndup(val, curr - val);
454 "ERROR: invalid fo_conf object passed to fo_config_is_list");
456 if ((tree = g_tree_lookup(conf->
group_map, group)) == NULL)
461 "ERROR: unknown group \"%s\"", group);
463 if ((val = g_tree_lookup(tree, key)) == NULL)
468 "ERROR: unknown key/value expression \"%s\"", key);
470 return val[0] ==
'[';
491 "ERROR: %s[%s] must be of type list to get length", group, key);
496 g_tree_lookup(conf->
group_map, group), key);
498 for (curr = val; *curr; curr++)
543 for (i = 0; i < ngroups; i++)
548 "Cannot join configuration with conflicting group \"%s\"", groups[i]);
554 for (i = 0; i < ngroups; i++)
556 keys = g_tree_lookup(src->
group_map, groups[i]);
557 keys = g_tree_ref(keys);
558 g_tree_insert(dst->
group_map, g_strdup(groups[i]), keys);
597 *length = g_tree_nnodes(conf->
group_map);
634 if ((tree = g_tree_lookup(conf->
group_map, group)) == NULL)
636 *length = g_tree_nnodes(tree);
638 if ((ret = g_tree_lookup(conf->
key_sets, group)))
643 g_tree_foreach(tree, (GTraverseFunc)
collect_keys, ret);
644 g_tree_insert(conf->
key_sets, g_strdup(group), ret);
662 return g_tree_lookup(conf->
group_map, group) != NULL;
681 if ((tree = g_tree_lookup(conf->
group_map, group)) == NULL)
683 return g_tree_lookup(tree, key) != NULL;
697 if (ptext && ptext[0])
699 int len = strlen(ptext);
700 while (isspace(ptext[len - 1])) ptext[--len] = 0;
701 while (isspace(*ptext)) ++ptext;
void fo_config_join(fo_conf *dst, fo_conf *src, GError **error)
Takes all groups and key from a fo_conf and adds them to another.
static gboolean fo_config_sub(const GMatchInfo *match, GString *ret, gpointer data)
int fo_config_has_group(fo_conf *conf, char *group)
Checks if the currently parsed configuration file has a specific group.
int fo_config_is_list(fo_conf *conf, char *group, char *key, GError **error)
Checks if a particular value is a list or just a normal value.
FOSSology library to read config file.
char ** group_set
Array of groups.
static const gchar * fo_conf_variable
Store the results of a regex match.
GTree * key_sets
Tree of sets of keys.
GRegex * fo_conf_parse
Regex for parsing.
GTree * group_map
Tree of groups in conf file.
static gboolean collect_keys(char *key, gpointer *value, char **data)
char * fo_config_get(fo_conf *conf, const char *group, const char *key, GError **error)
Gets an element based on its group name and key name. If the group or key is not found, the error object is set and NULL is returned.
char ** fo_config_group_set(fo_conf *conf, int *length)
Gets the set of group names.
static int fo_config_key(GTree *group, gchar *key, gchar *val, gboolean list, gchar *fname, guint line, GError **error)
Inserts a new Key/Value pair into the mapping of keys to values.
int fo_config_has_key(fo_conf *conf, char *group, char *key)
Checks if the a specific group in the currently parsed configuration file has a specific key...
#define BUFFER_SIZE
Maximum buffer length.
fo_conf * conf
The loaded configuration data.
Required group is missing.
static gint str_comp(gconstpointer a, gconstpointer b, gpointer user_data)
Requested group is invalid.
char * fo_config_get_list(fo_conf *conf, char *group, char *key, int idx, GError **error)
int fo_config_list_length(fo_conf *conf, char *group, char *key, GError **error)
Gets the length of the list associated with a particular list key.
Requested key is invalid.
void fo_config_free(fo_conf *conf)
Frees the memory associated with the internal configuration data structures.
list_t type structure used to keep various lists. (e.g. there are multiple lists).
char ** fo_config_key_set(fo_conf *conf, char *group, int *length)
Gets the set of key names for a particular group.
int n_groups
Number of groups.
#define throw_error(error, domain, code,...)
fo_conf * fo_config_load(char *rawname, GError **error)
Load the configuration information from the provided file.
GRegex * fo_conf_replace
Regex for replace.
static gboolean fo_config_eval(const GMatchInfo *match, GTree **g_current, fo_conf *dest, gchar *yyfile, guint yyline, GError **error)
Decides what to do with any one line of an input file.
static const gchar * fo_conf_pattern
char * trim(char *ptext)
Trimming whitespace.