868 lines
19 KiB
C
868 lines
19 KiB
C
/*
|
|
* Copyright (c) 2013 Hugh Bailey <obs.jim@gmail.com>
|
|
*
|
|
* Permission to use, copy, modify, and distribute this software for any
|
|
* purpose with or without fee is hereby granted, provided that the above
|
|
* copyright notice and this permission notice appear in all copies.
|
|
*
|
|
* THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
|
|
* WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
|
|
* MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
|
|
* ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
|
|
* WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
|
|
* ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
|
|
* OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
|
|
*/
|
|
|
|
#include <inttypes.h>
|
|
#include <stdio.h>
|
|
#include <wchar.h>
|
|
#include "config-file.h"
|
|
#include "threading.h"
|
|
#include "platform.h"
|
|
#include "base.h"
|
|
#include "bmem.h"
|
|
#include "darray.h"
|
|
#include "lexer.h"
|
|
#include "dstr.h"
|
|
|
|
struct config_item {
|
|
char *name;
|
|
char *value;
|
|
};
|
|
|
|
static inline void config_item_free(struct config_item *item)
|
|
{
|
|
bfree(item->name);
|
|
bfree(item->value);
|
|
}
|
|
|
|
struct config_section {
|
|
char *name;
|
|
struct darray items; /* struct config_item */
|
|
};
|
|
|
|
static inline void config_section_free(struct config_section *section)
|
|
{
|
|
struct config_item *items = section->items.array;
|
|
size_t i;
|
|
|
|
for (i = 0; i < section->items.num; i++)
|
|
config_item_free(items+i);
|
|
|
|
darray_free(§ion->items);
|
|
bfree(section->name);
|
|
}
|
|
|
|
struct config_data {
|
|
char *file;
|
|
struct darray sections; /* struct config_section */
|
|
struct darray defaults; /* struct config_section */
|
|
pthread_mutex_t mutex;
|
|
};
|
|
|
|
static inline bool init_mutex(config_t *config)
|
|
{
|
|
pthread_mutexattr_t attr;
|
|
if (pthread_mutexattr_init(&attr) != 0)
|
|
return false;
|
|
if (pthread_mutexattr_settype(&attr, PTHREAD_MUTEX_RECURSIVE) != 0)
|
|
return false;
|
|
return pthread_mutex_init(&config->mutex, &attr) == 0;
|
|
}
|
|
|
|
config_t *config_create(const char *file)
|
|
{
|
|
struct config_data *config;
|
|
FILE *f;
|
|
|
|
f = os_fopen(file, "wb");
|
|
if (!f)
|
|
return NULL;
|
|
fclose(f);
|
|
|
|
config = bzalloc(sizeof(struct config_data));
|
|
|
|
if (!init_mutex(config)) {
|
|
bfree(config);
|
|
return NULL;
|
|
}
|
|
|
|
config->file = bstrdup(file);
|
|
return config;
|
|
}
|
|
|
|
static inline void remove_ref_whitespace(struct strref *ref)
|
|
{
|
|
if (ref->array) {
|
|
while (is_whitespace(*ref->array)) {
|
|
ref->array++;
|
|
ref->len--;
|
|
}
|
|
|
|
while (ref->len && is_whitespace(ref->array[ref->len-1]))
|
|
ref->len--;
|
|
}
|
|
}
|
|
|
|
static bool config_parse_string(struct lexer *lex, struct strref *ref,
|
|
char end)
|
|
{
|
|
bool success = end != 0;
|
|
struct base_token token;
|
|
base_token_clear(&token);
|
|
|
|
while (lexer_getbasetoken(lex, &token, PARSE_WHITESPACE)) {
|
|
if (end) {
|
|
if (*token.text.array == end) {
|
|
success = true;
|
|
break;
|
|
} else if (is_newline(*token.text.array)) {
|
|
success = false;
|
|
break;
|
|
}
|
|
} else {
|
|
if (is_newline(*token.text.array)) {
|
|
success = true;
|
|
break;
|
|
}
|
|
}
|
|
|
|
strref_add(ref, &token.text);
|
|
}
|
|
|
|
remove_ref_whitespace(ref);
|
|
return success;
|
|
}
|
|
|
|
static void unescape(struct dstr *str)
|
|
{
|
|
char *read = str->array;
|
|
char *write = str->array;
|
|
|
|
for (; *read; read++, write++) {
|
|
char cur = *read;
|
|
if (cur == '\\') {
|
|
char next = read[1];
|
|
if (next == '\\') {
|
|
read++;
|
|
} else if (next == 'r') {
|
|
cur = '\r';
|
|
read++;
|
|
} else if (next =='n') {
|
|
cur = '\n';
|
|
read++;
|
|
}
|
|
}
|
|
|
|
if (read != write)
|
|
*write = cur;
|
|
}
|
|
|
|
if (read != write)
|
|
*write = '\0';
|
|
}
|
|
|
|
static void config_add_item(struct darray *items, struct strref *name,
|
|
struct strref *value)
|
|
{
|
|
struct config_item item;
|
|
struct dstr item_value;
|
|
dstr_init_copy_strref(&item_value, value);
|
|
|
|
unescape(&item_value);
|
|
|
|
item.name = bstrdup_n(name->array, name->len);
|
|
item.value = item_value.array;
|
|
darray_push_back(sizeof(struct config_item), items, &item);
|
|
}
|
|
|
|
static void config_parse_section(struct config_section *section,
|
|
struct lexer *lex)
|
|
{
|
|
struct base_token token;
|
|
|
|
while (lexer_getbasetoken(lex, &token, PARSE_WHITESPACE)) {
|
|
struct strref name, value;
|
|
|
|
while (token.type == BASETOKEN_WHITESPACE) {
|
|
if (!lexer_getbasetoken(lex, &token, PARSE_WHITESPACE))
|
|
return;
|
|
}
|
|
|
|
if (token.type == BASETOKEN_OTHER) {
|
|
if (*token.text.array == '#') {
|
|
do {
|
|
if (!lexer_getbasetoken(lex, &token,
|
|
PARSE_WHITESPACE))
|
|
return;
|
|
} while (!is_newline(*token.text.array));
|
|
|
|
continue;
|
|
} else if (*token.text.array == '[') {
|
|
lex->offset--;
|
|
return;
|
|
}
|
|
}
|
|
|
|
strref_copy(&name, &token.text);
|
|
if (!config_parse_string(lex, &name, '='))
|
|
continue;
|
|
|
|
strref_clear(&value);
|
|
config_parse_string(lex, &value, 0);
|
|
|
|
if (strref_is_empty(&value)) {
|
|
struct config_item item;
|
|
item.name = bstrdup_n(name.array, name.len);
|
|
item.value = bzalloc(1);
|
|
darray_push_back(sizeof(struct config_item),
|
|
§ion->items, &item);
|
|
} else {
|
|
config_add_item(§ion->items, &name, &value);
|
|
}
|
|
}
|
|
}
|
|
|
|
static void parse_config_data(struct darray *sections, struct lexer *lex)
|
|
{
|
|
struct strref section_name;
|
|
struct base_token token;
|
|
|
|
base_token_clear(&token);
|
|
|
|
while (lexer_getbasetoken(lex, &token, PARSE_WHITESPACE)) {
|
|
struct config_section *section;
|
|
|
|
while (token.type == BASETOKEN_WHITESPACE) {
|
|
if (!lexer_getbasetoken(lex, &token, PARSE_WHITESPACE))
|
|
return;
|
|
}
|
|
|
|
if (*token.text.array != '[') {
|
|
while (!is_newline(*token.text.array)) {
|
|
if (!lexer_getbasetoken(lex, &token,
|
|
PARSE_WHITESPACE))
|
|
return;
|
|
}
|
|
|
|
continue;
|
|
}
|
|
|
|
strref_clear(§ion_name);
|
|
config_parse_string(lex, §ion_name, ']');
|
|
if (!section_name.len)
|
|
return;
|
|
|
|
section = darray_push_back_new(sizeof(struct config_section),
|
|
sections);
|
|
section->name = bstrdup_n(section_name.array,
|
|
section_name.len);
|
|
config_parse_section(section, lex);
|
|
}
|
|
}
|
|
|
|
static int config_parse_file(struct darray *sections, const char *file,
|
|
bool always_open)
|
|
{
|
|
char *file_data;
|
|
struct lexer lex;
|
|
FILE *f;
|
|
|
|
f = os_fopen(file, "rb");
|
|
if (always_open && !f)
|
|
f = os_fopen(file, "w+");
|
|
if (!f)
|
|
return CONFIG_FILENOTFOUND;
|
|
|
|
os_fread_utf8(f, &file_data);
|
|
fclose(f);
|
|
|
|
if (!file_data)
|
|
return CONFIG_SUCCESS;
|
|
|
|
lexer_init(&lex);
|
|
lexer_start_move(&lex, file_data);
|
|
|
|
parse_config_data(sections, &lex);
|
|
|
|
lexer_free(&lex);
|
|
return CONFIG_SUCCESS;
|
|
}
|
|
|
|
int config_open(config_t **config, const char *file,
|
|
enum config_open_type open_type)
|
|
{
|
|
int errorcode;
|
|
bool always_open = open_type == CONFIG_OPEN_ALWAYS;
|
|
|
|
if (!config)
|
|
return CONFIG_ERROR;
|
|
|
|
*config = bzalloc(sizeof(struct config_data));
|
|
if (!*config)
|
|
return CONFIG_ERROR;
|
|
|
|
if (!init_mutex(*config)) {
|
|
bfree(*config);
|
|
return CONFIG_ERROR;
|
|
}
|
|
|
|
(*config)->file = bstrdup(file);
|
|
|
|
errorcode = config_parse_file(&(*config)->sections, file, always_open);
|
|
|
|
if (errorcode != CONFIG_SUCCESS) {
|
|
config_close(*config);
|
|
*config = NULL;
|
|
}
|
|
|
|
return errorcode;
|
|
}
|
|
|
|
int config_open_string(config_t **config, const char *str)
|
|
{
|
|
struct lexer lex;
|
|
|
|
if (!config)
|
|
return CONFIG_ERROR;
|
|
|
|
*config = bzalloc(sizeof(struct config_data));
|
|
if (!*config)
|
|
return CONFIG_ERROR;
|
|
|
|
if (!init_mutex(*config)) {
|
|
bfree(*config);
|
|
return CONFIG_ERROR;
|
|
}
|
|
|
|
(*config)->file = NULL;
|
|
|
|
lexer_init(&lex);
|
|
lexer_start(&lex, str);
|
|
parse_config_data(&(*config)->sections, &lex);
|
|
lexer_free(&lex);
|
|
|
|
return CONFIG_SUCCESS;
|
|
}
|
|
|
|
int config_open_defaults(config_t *config, const char *file)
|
|
{
|
|
if (!config)
|
|
return CONFIG_ERROR;
|
|
|
|
return config_parse_file(&config->defaults, file, false);
|
|
}
|
|
|
|
int config_save(config_t *config)
|
|
{
|
|
FILE *f;
|
|
struct dstr str, tmp;
|
|
size_t i, j;
|
|
int ret = CONFIG_ERROR;
|
|
|
|
if (!config)
|
|
return CONFIG_ERROR;
|
|
if (!config->file)
|
|
return CONFIG_ERROR;
|
|
|
|
dstr_init(&str);
|
|
dstr_init(&tmp);
|
|
|
|
pthread_mutex_lock(&config->mutex);
|
|
|
|
f = os_fopen(config->file, "wb");
|
|
if (!f) {
|
|
pthread_mutex_unlock(&config->mutex);
|
|
return CONFIG_FILENOTFOUND;
|
|
}
|
|
|
|
for (i = 0; i < config->sections.num; i++) {
|
|
struct config_section *section = darray_item(
|
|
sizeof(struct config_section),
|
|
&config->sections, i);
|
|
|
|
if (i) dstr_cat(&str, "\n");
|
|
|
|
dstr_cat(&str, "[");
|
|
dstr_cat(&str, section->name);
|
|
dstr_cat(&str, "]\n");
|
|
|
|
for (j = 0; j < section->items.num; j++) {
|
|
struct config_item *item = darray_item(
|
|
sizeof(struct config_item),
|
|
§ion->items, j);
|
|
|
|
dstr_copy(&tmp, item->value ? item->value : "");
|
|
dstr_replace(&tmp, "\\", "\\\\");
|
|
dstr_replace(&tmp, "\r", "\\r");
|
|
dstr_replace(&tmp, "\n", "\\n");
|
|
|
|
dstr_cat(&str, item->name);
|
|
dstr_cat(&str, "=");
|
|
dstr_cat(&str, tmp.array);
|
|
dstr_cat(&str, "\n");
|
|
}
|
|
}
|
|
|
|
#ifdef _WIN32
|
|
if (fwrite("\xEF\xBB\xBF", 3, 1, f) != 1)
|
|
goto cleanup;
|
|
#endif
|
|
if (fwrite(str.array, str.len, 1, f) != 1)
|
|
goto cleanup;
|
|
|
|
ret = CONFIG_SUCCESS;
|
|
|
|
cleanup:
|
|
fclose(f);
|
|
|
|
pthread_mutex_unlock(&config->mutex);
|
|
|
|
dstr_free(&tmp);
|
|
dstr_free(&str);
|
|
|
|
return ret;
|
|
}
|
|
|
|
int config_save_safe(config_t *config, const char *temp_ext,
|
|
const char *backup_ext)
|
|
{
|
|
struct dstr temp_file = {0};
|
|
struct dstr backup_file = {0};
|
|
char *file = config->file;
|
|
int ret;
|
|
|
|
if (!temp_ext || !*temp_ext) {
|
|
blog(LOG_ERROR, "config_save_safe: invalid "
|
|
"temporary extension specified");
|
|
return CONFIG_ERROR;
|
|
}
|
|
|
|
pthread_mutex_lock(&config->mutex);
|
|
|
|
dstr_copy(&temp_file, config->file);
|
|
if (*temp_ext != '.')
|
|
dstr_cat(&temp_file, ".");
|
|
dstr_cat(&temp_file, temp_ext);
|
|
|
|
config->file = temp_file.array;
|
|
ret = config_save(config);
|
|
config->file = file;
|
|
|
|
if (ret != CONFIG_SUCCESS) {
|
|
blog(LOG_ERROR, "config_save_safe: failed to "
|
|
"write to %s", temp_file.array);
|
|
goto cleanup;
|
|
}
|
|
|
|
if (backup_ext && *backup_ext) {
|
|
dstr_copy(&backup_file, config->file);
|
|
if (*backup_ext != '.')
|
|
dstr_cat(&backup_file, ".");
|
|
dstr_cat(&backup_file, backup_ext);
|
|
}
|
|
|
|
if (os_safe_replace(file, temp_file.array, backup_file.array) != 0)
|
|
ret = CONFIG_ERROR;
|
|
|
|
cleanup:
|
|
pthread_mutex_unlock(&config->mutex);
|
|
dstr_free(&temp_file);
|
|
dstr_free(&backup_file);
|
|
return ret;
|
|
}
|
|
|
|
void config_close(config_t *config)
|
|
{
|
|
struct config_section *defaults, *sections;
|
|
size_t i;
|
|
|
|
if (!config) return;
|
|
|
|
defaults = config->defaults.array;
|
|
sections = config->sections.array;
|
|
|
|
for (i = 0; i < config->defaults.num; i++)
|
|
config_section_free(defaults+i);
|
|
for (i = 0; i < config->sections.num; i++)
|
|
config_section_free(sections+i);
|
|
|
|
darray_free(&config->defaults);
|
|
darray_free(&config->sections);
|
|
bfree(config->file);
|
|
pthread_mutex_destroy(&config->mutex);
|
|
bfree(config);
|
|
}
|
|
|
|
size_t config_num_sections(config_t *config)
|
|
{
|
|
return config->sections.num;
|
|
}
|
|
|
|
const char *config_get_section(config_t *config, size_t idx)
|
|
{
|
|
struct config_section *section;
|
|
const char *name = NULL;
|
|
|
|
pthread_mutex_lock(&config->mutex);
|
|
|
|
if (idx >= config->sections.num)
|
|
goto unlock;
|
|
|
|
section = darray_item(sizeof(struct config_section), &config->sections,
|
|
idx);
|
|
name = section->name;
|
|
|
|
unlock:
|
|
pthread_mutex_unlock(&config->mutex);
|
|
return name;
|
|
}
|
|
|
|
static const struct config_item *config_find_item(const struct darray *sections,
|
|
const char *section, const char *name)
|
|
{
|
|
size_t i, j;
|
|
|
|
for (i = 0; i < sections->num; i++) {
|
|
const struct config_section *sec = darray_item(
|
|
sizeof(struct config_section), sections, i);
|
|
|
|
if (astrcmpi(sec->name, section) == 0) {
|
|
for (j = 0; j < sec->items.num; j++) {
|
|
struct config_item *item = darray_item(
|
|
sizeof(struct config_item),
|
|
&sec->items, j);
|
|
|
|
if (astrcmpi(item->name, name) == 0)
|
|
return item;
|
|
}
|
|
}
|
|
}
|
|
|
|
return NULL;
|
|
}
|
|
|
|
static void config_set_item(config_t *config, struct darray *sections,
|
|
const char *section, const char *name, char *value)
|
|
{
|
|
struct config_section *sec = NULL;
|
|
struct config_section *array = sections->array;
|
|
struct config_item *item;
|
|
size_t i, j;
|
|
|
|
pthread_mutex_lock(&config->mutex);
|
|
|
|
for (i = 0; i < sections->num; i++) {
|
|
struct config_section *cur_sec = array+i;
|
|
struct config_item *items = cur_sec->items.array;
|
|
|
|
if (astrcmpi(cur_sec->name, section) == 0) {
|
|
for (j = 0; j < cur_sec->items.num; j++) {
|
|
item = items+j;
|
|
|
|
if (astrcmpi(item->name, name) == 0) {
|
|
bfree(item->value);
|
|
item->value = value;
|
|
goto unlock;
|
|
}
|
|
}
|
|
|
|
sec = cur_sec;
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (!sec) {
|
|
sec = darray_push_back_new(sizeof(struct config_section),
|
|
sections);
|
|
sec->name = bstrdup(section);
|
|
}
|
|
|
|
item = darray_push_back_new(sizeof(struct config_item), &sec->items);
|
|
item->name = bstrdup(name);
|
|
item->value = value;
|
|
|
|
unlock:
|
|
pthread_mutex_unlock(&config->mutex);
|
|
}
|
|
|
|
void config_set_string(config_t *config, const char *section,
|
|
const char *name, const char *value)
|
|
{
|
|
if (!value)
|
|
value = "";
|
|
config_set_item(config, &config->sections, section, name,
|
|
bstrdup(value));
|
|
}
|
|
|
|
void config_set_int(config_t *config, const char *section,
|
|
const char *name, int64_t value)
|
|
{
|
|
struct dstr str;
|
|
dstr_init(&str);
|
|
dstr_printf(&str, "%"PRId64, value);
|
|
config_set_item(config, &config->sections, section, name, str.array);
|
|
}
|
|
|
|
void config_set_uint(config_t *config, const char *section,
|
|
const char *name, uint64_t value)
|
|
{
|
|
struct dstr str;
|
|
dstr_init(&str);
|
|
dstr_printf(&str, "%"PRIu64, value);
|
|
config_set_item(config, &config->sections, section, name, str.array);
|
|
}
|
|
|
|
void config_set_bool(config_t *config, const char *section,
|
|
const char *name, bool value)
|
|
{
|
|
char *str = bstrdup(value ? "true" : "false");
|
|
config_set_item(config, &config->sections, section, name, str);
|
|
}
|
|
|
|
void config_set_double(config_t *config, const char *section,
|
|
const char *name, double value)
|
|
{
|
|
char *str = bzalloc(64);
|
|
os_dtostr(value, str, 64);
|
|
config_set_item(config, &config->sections, section, name, str);
|
|
}
|
|
|
|
void config_set_default_string(config_t *config, const char *section,
|
|
const char *name, const char *value)
|
|
{
|
|
if (!value)
|
|
value = "";
|
|
config_set_item(config, &config->defaults, section, name,
|
|
bstrdup(value));
|
|
}
|
|
|
|
void config_set_default_int(config_t *config, const char *section,
|
|
const char *name, int64_t value)
|
|
{
|
|
struct dstr str;
|
|
dstr_init(&str);
|
|
dstr_printf(&str, "%"PRId64, value);
|
|
config_set_item(config, &config->defaults, section, name, str.array);
|
|
}
|
|
|
|
void config_set_default_uint(config_t *config, const char *section,
|
|
const char *name, uint64_t value)
|
|
{
|
|
struct dstr str;
|
|
dstr_init(&str);
|
|
dstr_printf(&str, "%"PRIu64, value);
|
|
config_set_item(config, &config->defaults, section, name, str.array);
|
|
}
|
|
|
|
void config_set_default_bool(config_t *config, const char *section,
|
|
const char *name, bool value)
|
|
{
|
|
char *str = bstrdup(value ? "true" : "false");
|
|
config_set_item(config, &config->defaults, section, name, str);
|
|
}
|
|
|
|
void config_set_default_double(config_t *config, const char *section,
|
|
const char *name, double value)
|
|
{
|
|
struct dstr str;
|
|
dstr_init(&str);
|
|
dstr_printf(&str, "%g", value);
|
|
config_set_item(config, &config->defaults, section, name, str.array);
|
|
}
|
|
|
|
const char *config_get_string(config_t *config, const char *section,
|
|
const char *name)
|
|
{
|
|
const struct config_item *item;
|
|
const char *value = NULL;
|
|
|
|
pthread_mutex_lock(&config->mutex);
|
|
|
|
item = config_find_item(&config->sections, section, name);
|
|
if (!item)
|
|
item = config_find_item(&config->defaults, section, name);
|
|
if (item)
|
|
value = item->value;
|
|
|
|
pthread_mutex_unlock(&config->mutex);
|
|
return value;
|
|
}
|
|
|
|
static inline int64_t str_to_int64(const char *str)
|
|
{
|
|
if (!str || !*str)
|
|
return 0;
|
|
|
|
if (str[0] == '0' && str[1] == 'x')
|
|
return strtoll(str + 2, NULL, 16);
|
|
else
|
|
return strtoll(str, NULL, 10);
|
|
}
|
|
|
|
static inline uint64_t str_to_uint64(const char *str)
|
|
{
|
|
if (!str || !*str)
|
|
return 0;
|
|
|
|
if (str[0] == '0' && str[1] == 'x')
|
|
return strtoull(str + 2, NULL, 16);
|
|
else
|
|
return strtoull(str, NULL, 10);
|
|
}
|
|
|
|
int64_t config_get_int(config_t *config, const char *section,
|
|
const char *name)
|
|
{
|
|
const char *value = config_get_string(config, section, name);
|
|
if (value)
|
|
return str_to_int64(value);
|
|
|
|
return 0;
|
|
}
|
|
|
|
uint64_t config_get_uint(config_t *config, const char *section,
|
|
const char *name)
|
|
{
|
|
const char *value = config_get_string(config, section, name);
|
|
if (value)
|
|
return str_to_uint64(value);
|
|
|
|
return 0;
|
|
}
|
|
|
|
bool config_get_bool(config_t *config, const char *section,
|
|
const char *name)
|
|
{
|
|
const char *value = config_get_string(config, section, name);
|
|
if (value)
|
|
return astrcmpi(value, "true") == 0 ||
|
|
!!str_to_uint64(value);
|
|
|
|
return false;
|
|
}
|
|
|
|
double config_get_double(config_t *config, const char *section,
|
|
const char *name)
|
|
{
|
|
const char *value = config_get_string(config, section, name);
|
|
if (value)
|
|
return os_strtod(value);
|
|
|
|
return 0.0;
|
|
}
|
|
|
|
bool config_remove_value(config_t *config, const char *section,
|
|
const char *name)
|
|
{
|
|
struct darray *sections = &config->sections;
|
|
bool success = false;
|
|
|
|
pthread_mutex_lock(&config->mutex);
|
|
|
|
for (size_t i = 0; i < sections->num; i++) {
|
|
struct config_section *sec = darray_item(
|
|
sizeof(struct config_section), sections, i);
|
|
|
|
if (astrcmpi(sec->name, section) != 0)
|
|
continue;
|
|
|
|
for (size_t j = 0; j < sec->items.num; j++) {
|
|
struct config_item *item = darray_item(
|
|
sizeof(struct config_item),
|
|
&sec->items, j);
|
|
|
|
if (astrcmpi(item->name, name) == 0) {
|
|
config_item_free(item);
|
|
darray_erase(sizeof(struct config_item),
|
|
&sec->items, j);
|
|
success = true;
|
|
goto unlock;
|
|
}
|
|
}
|
|
}
|
|
|
|
unlock:
|
|
pthread_mutex_unlock(&config->mutex);
|
|
return success;
|
|
}
|
|
|
|
const char *config_get_default_string(config_t *config,
|
|
const char *section, const char *name)
|
|
{
|
|
const struct config_item *item;
|
|
const char *value = NULL;
|
|
|
|
pthread_mutex_lock(&config->mutex);
|
|
|
|
item = config_find_item(&config->defaults, section, name);
|
|
if (item)
|
|
value = item->value;
|
|
|
|
pthread_mutex_unlock(&config->mutex);
|
|
return value;
|
|
}
|
|
|
|
int64_t config_get_default_int(config_t *config, const char *section,
|
|
const char *name)
|
|
{
|
|
const char *value = config_get_default_string(config, section, name);
|
|
if (value)
|
|
return str_to_int64(value);
|
|
|
|
return 0;
|
|
}
|
|
|
|
uint64_t config_get_default_uint(config_t *config, const char *section,
|
|
const char *name)
|
|
{
|
|
const char *value = config_get_default_string(config, section, name);
|
|
if (value)
|
|
return str_to_uint64(value);
|
|
|
|
return 0;
|
|
}
|
|
|
|
bool config_get_default_bool(config_t *config, const char *section,
|
|
const char *name)
|
|
{
|
|
const char *value = config_get_default_string(config, section, name);
|
|
if (value)
|
|
return astrcmpi(value, "true") == 0 ||
|
|
!!str_to_uint64(value);
|
|
|
|
return false;
|
|
}
|
|
|
|
double config_get_default_double(config_t *config, const char *section,
|
|
const char *name)
|
|
{
|
|
const char *value = config_get_default_string(config, section, name);
|
|
if (value)
|
|
return os_strtod(value);
|
|
|
|
return 0.0;
|
|
}
|
|
|
|
bool config_has_user_value(config_t *config, const char *section,
|
|
const char *name)
|
|
{
|
|
bool success;
|
|
pthread_mutex_lock(&config->mutex);
|
|
success = config_find_item(&config->sections, section, name) != NULL;
|
|
pthread_mutex_unlock(&config->mutex);
|
|
return success;
|
|
}
|
|
|
|
bool config_has_default_value(config_t *config, const char *section,
|
|
const char *name)
|
|
{
|
|
bool success;
|
|
pthread_mutex_lock(&config->mutex);
|
|
success = config_find_item(&config->defaults, section, name) != NULL;
|
|
pthread_mutex_unlock(&config->mutex);
|
|
return success;
|
|
}
|
|
|