2018-05-08 17:04:49 +03:00
|
|
|
/* vim: tabstop=4 shiftwidth=4 noexpandtab
|
|
|
|
* This file is part of ToaruOS and is released under the terms
|
|
|
|
* of the NCSA / University of Illinois License - see LICENSE.md
|
|
|
|
* Copyright (C) 2018 K. Lange
|
|
|
|
*
|
|
|
|
* Configuration File Reader
|
|
|
|
*
|
|
|
|
* Reads an implementation of the INI "standard". Note that INI
|
|
|
|
* isn't actually a standard. We support the following:
|
|
|
|
* - ; comments
|
|
|
|
* - foo=bar keyword assignment
|
|
|
|
* - [sections]
|
|
|
|
*/
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <toaru/hashmap.h>
|
|
|
|
#include <toaru/confreader.h>
|
|
|
|
|
|
|
|
#define TRACE_APP_NAME "confreader"
|
|
|
|
#define TRACE(...)
|
|
|
|
//#include <toaru/trace.h>
|
|
|
|
|
|
|
|
static void free_hashmap(void * h) {
|
|
|
|
hashmap_free(h);
|
|
|
|
free(h);
|
|
|
|
}
|
|
|
|
|
2018-11-12 03:26:30 +03:00
|
|
|
static int write_section(FILE * f, hashmap_t * section) {
|
|
|
|
list_t * keys = hashmap_keys(section);
|
|
|
|
foreach(node, keys) {
|
|
|
|
char * key = (char*)node->value;
|
|
|
|
char * value = hashmap_get(section, key);
|
|
|
|
fprintf(f, "%s=%s\n", key, value);
|
|
|
|
}
|
|
|
|
list_free(keys);
|
|
|
|
free(keys);
|
|
|
|
return 0;
|
|
|
|
}
|
2018-05-08 17:04:49 +03:00
|
|
|
|
2018-11-12 03:26:30 +03:00
|
|
|
int confreader_write(confreader_t * config, const char * file) {
|
|
|
|
|
|
|
|
FILE * f = fopen(file, "w");
|
|
|
|
|
|
|
|
if (!f) return 1;
|
|
|
|
|
|
|
|
hashmap_t * base = hashmap_get(config->sections, "");
|
|
|
|
if (base) {
|
|
|
|
write_section(f, base);
|
|
|
|
}
|
|
|
|
|
|
|
|
list_t * sections = hashmap_keys(config->sections);
|
|
|
|
foreach(node, sections) {
|
|
|
|
char * section = (char*)node->value;
|
|
|
|
if (strcmp(section,"")) {
|
|
|
|
hashmap_t * data = hashmap_get(config->sections, section);
|
|
|
|
fprintf(f, "[%s]\n", section);
|
|
|
|
write_section(f, data);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
2018-05-08 17:04:49 +03:00
|
|
|
|
2018-11-12 03:26:30 +03:00
|
|
|
confreader_t * confreader_create_empty(void) {
|
|
|
|
confreader_t * out = malloc(sizeof(confreader_t));
|
2018-05-08 17:04:49 +03:00
|
|
|
out->sections = hashmap_create(10);
|
2018-11-12 03:26:30 +03:00
|
|
|
return out;
|
|
|
|
}
|
|
|
|
|
|
|
|
confreader_t * confreader_load(const char * file) {
|
2018-05-08 17:04:49 +03:00
|
|
|
|
|
|
|
FILE * f = fopen(file, "r");
|
|
|
|
|
2018-11-12 03:26:30 +03:00
|
|
|
if (!f) return NULL;
|
|
|
|
|
|
|
|
confreader_t * out = confreader_create_empty();
|
|
|
|
|
2018-05-08 17:04:49 +03:00
|
|
|
hashmap_t * current_section = hashmap_create(10);
|
|
|
|
current_section->hash_val_free = free_hashmap;
|
|
|
|
|
|
|
|
hashmap_set(out->sections, "", current_section);
|
|
|
|
|
|
|
|
if (!f) {
|
|
|
|
/* File does not exist, no configuration values, but continue normally. */
|
|
|
|
return out;
|
|
|
|
}
|
|
|
|
|
|
|
|
char tmp[1024];
|
|
|
|
char tmp2[1024];
|
|
|
|
|
|
|
|
while (!feof(f)) {
|
|
|
|
char c = fgetc(f);
|
|
|
|
tmp[0] = '\0';
|
|
|
|
tmp2[0] = '\0';
|
|
|
|
if (c == ';') {
|
|
|
|
TRACE("Comment");
|
|
|
|
while (!feof(f) && fgetc(f) != '\n');
|
|
|
|
TRACE("Done");
|
|
|
|
} else if (c == '\n' || c == EOF) {
|
|
|
|
TRACE("blank line or EOF: %d", c);
|
|
|
|
continue;
|
|
|
|
} else if (c == '[') {
|
|
|
|
TRACE("section");
|
|
|
|
char * foo = tmp;
|
|
|
|
int i;
|
|
|
|
while ((i = fgetc(f)) >= 0) {
|
|
|
|
if (i == ']') break;
|
|
|
|
*foo = i;
|
|
|
|
foo++;
|
|
|
|
*foo = '\0';
|
|
|
|
}
|
|
|
|
while (!feof(f) && fgetc(f) != '\n');
|
|
|
|
current_section = hashmap_create(10);
|
|
|
|
TRACE("adding section %s", tmp);
|
|
|
|
hashmap_set(out->sections, tmp, current_section);
|
|
|
|
TRACE("section is over");
|
|
|
|
} else {
|
|
|
|
TRACE("value");
|
|
|
|
char * foo = tmp;
|
|
|
|
*foo = c;
|
|
|
|
foo++;
|
|
|
|
int i;
|
|
|
|
while ((i = fgetc(f)) >= 0) {
|
|
|
|
if (i == '=') break;
|
|
|
|
*foo = i;
|
|
|
|
foo++;
|
|
|
|
*foo = '\0';
|
|
|
|
}
|
|
|
|
if (i != '=') {
|
|
|
|
TRACE("no equals sign");
|
|
|
|
while (!feof(f) && fgetc(f) != '\n');
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
TRACE("=");
|
|
|
|
foo = tmp2;
|
|
|
|
while ((i = fgetc(f)) >= 0) {
|
|
|
|
if (i == '\n') break;
|
|
|
|
*foo = i;
|
|
|
|
foo++;
|
|
|
|
*foo = '\0';
|
|
|
|
}
|
|
|
|
TRACE("setting value %s to %s", tmp, tmp2);
|
|
|
|
hashmap_set(current_section, tmp, strdup(tmp2));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fclose(f);
|
|
|
|
|
|
|
|
TRACE("done reading");
|
|
|
|
|
|
|
|
return out;
|
|
|
|
}
|
|
|
|
|
|
|
|
void confreader_free(confreader_t * conf) {
|
|
|
|
free_hashmap(conf->sections);
|
|
|
|
free(conf);
|
|
|
|
}
|
|
|
|
|
|
|
|
char * confreader_get(confreader_t * ctx, char * section, char * value) {
|
|
|
|
if (!ctx) return NULL;
|
|
|
|
|
|
|
|
hashmap_t * s = hashmap_get(ctx->sections, section);
|
|
|
|
|
|
|
|
if (!s) {
|
|
|
|
TRACE("section doesn't exist: %s", section);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
return hashmap_get(s, value);
|
|
|
|
}
|
|
|
|
|
|
|
|
char * confreader_getd(confreader_t * ctx, char * section, char * value, char * def) {
|
|
|
|
char * result = confreader_get(ctx, section, value);
|
|
|
|
return result ? result : def;
|
|
|
|
}
|
|
|
|
|
|
|
|
int confreader_int(confreader_t * ctx, char * section, char * value) {
|
|
|
|
char * result = confreader_get(ctx, section, value);
|
|
|
|
if (!result) return 0;
|
|
|
|
return atoi(result);
|
|
|
|
}
|
|
|
|
|
|
|
|
int confreader_intd(confreader_t * ctx, char * section, char * value, int def) {
|
|
|
|
char * result = confreader_get(ctx, section, value);
|
|
|
|
if (!result) return def;
|
|
|
|
return atoi(result);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|