/* * Copyright (c) 2013 Hugh Bailey * * 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 #include #include #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; }