/* config-ini.c -- INI config file parser * This file is part of redshift-ng. * * redshift-ng is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. * * redshift-ng is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with redshift-ng. If not, see . * * Copyright (c) 2010-2018 Jon Lund Steffensen * Copyright (c) 2025 Mattias Andrée */ #include "common.h" /** * Specification for a path that consists of a two parts: * the first being defined by the environment, and the * seocnd being a static string */ struct env_path { /** * Whether the environment variable referenced by `.prefix` * should be split at each colon (:) into multiple paths to * test */ int multidir_env; /** * Environment variable to use as the first part of the path * * `NULL` if the user's home directory should be used * * The empty string if `.suffix` should be used as is */ const char *prefix_env; /** * The second part of the path */ const char *suffix; }; /** * Paths, in order of priority, to test when looking for * the configuration file for redshift */ static const struct env_path paths[] = { {0, "XDG_CONFIG_HOME", "/redshift-ng/redshift.conf"}, {0, "XDG_CONFIG_HOME", "/redshift/redshift.conf"}, {0, "XDG_CONFIG_HOME", "/redshift.conf"}, #ifdef WINDOWS {0, "localappdata", "/redshift-ng/redshift.conf"}, {0, "localappdata", "/redshift/redshift.conf"}, {0, "localappdata", "/redshift.conf"}, #endif {0, "HOME", "/.config/redshift-ng/redshift.conf"}, {0, "HOME", "/.config/redshift/redshift.conf"}, {0, "HOME", "/.config/redshift.conf"}, {0, "HOME", "/.redshift.conf"}, {0, NULL, "/.config/redshift-ng/redshift.conf"}, {0, NULL, "/.config/redshift/redshift.conf"}, {0, NULL, "/.config/redshift.conf"}, {0, NULL, "/.redshift.conf"}, {1, "XDG_CONFIG_DIRS", "/redshift-ng/redshift.conf"}, {1, "XDG_CONFIG_DIRS", "/redshift/redshift.conf"}, {1, "XDG_CONFIG_DIRS", "/redshift.conf"}, {0, "", "/etc/redshift-ng/redshift.conf"}, {0, "", "/etc/redshift/redshift.conf"}, {0, "", "/etc/redshift.conf"} }; /** * Remove trailing whitespace * * @param s The string to trim, will be truncated * @param end The current end of `s`; will be looked up if `NULL` * @return `s` */ static char * rtrim(char *s, char *end) { end = end ? end : strchr(s, '\0'); while (end != s && (end[-1] == ' ' || end[-1] == '\t')) end--; *end = '\0'; return s; } /** * Remove leading whitespace * * @param s The string to trim (will not be modified) * @return An offset of `s` */ GCC_ONLY(__attribute__((__warn_unused_result__))) static char * ltrim(char *s) { return &s[strspn(s, " \t")]; } /** * Get the user's home directory * * This function looks up the user's home directory * once and caches the result, later calls to this * function will use the cached result * * @param The user's home directory; the empty string if not found */ static const char * get_home(void) { #ifdef WINDOWS return NULL; #else static const char *home = NULL; struct passwd *pw; if (!home) { pw = getpwuid(getuid()); if (pw) { home = pw->pw_dir; if (home && *home) return home; weprintf(_("Cannot determine your home directory, " "it is from the system's user table.")); } else if (errno) { weprintf("getpwuid:"); } else { weprintf(_("Cannot determine your home directory, your" " user ID is missing from the system's user table.")); /* `errno` can either be set to any number of error codes, * or be zero if the user does not have a passwd entry */ } home = ""; } return home; #endif } /** * Search for a file and open it for reading * * @param path_spec Specification for the path to try * @param path_out Output parameter for the file path * @return `FILE` object for the reading the file; `NULL` if not found */ static FILE * try_path(const struct env_path *path_spec, char **path_out) { const char *prefix, *p, *q; char *path; size_t len; FILE *f; if (!path_spec->prefix_env) prefix = get_home(); else if (!*path_spec->prefix_env) return fopen(path_spec->suffix, "r"); else prefix = getenv(path_spec->prefix_env); if (!prefix || !*prefix) return NULL; path = emalloc(strlen(prefix) + strlen(path_spec->suffix) + 1U); if (path_spec->multidir_env) { for (p = prefix; *p; p = &q[!!*q]) { #ifdef strchrnul q = strchrnul(p, ':'); #else q = strchr(p, ':'); q = q ? q : strchr(p, '\0'); #endif len = (size_t)(q - p); if (!len) continue; memcpy(path, p, len); stpcpy(&path[len], path_spec->suffix); f = fopen(path, "r"); if (f) { weprintf(_("Found configuration file `%s'."), path); break; } else if (errno != ENOENT) { eprintf("fopen %s \"r\":", path); } } } else { stpcpy(stpcpy(path, prefix), path_spec->suffix); f = fopen(path, "r"); if (f) weprintf(_("Found configuration file `%s'."), path); else if (errno != ENOENT) eprintf("fopen %s \"r\":", path); } if (f) { *path_out = path; } else { free(path); *path_out = NULL; } return f; } /** * Open the configuration file for reading * * @param path The path to the configuration file, or `NULL` if the * application should look for it in the default paths * @param path_out Output parameter for the configuration file * @param pathbuf_out Output parameter for the memory allocation for `*path_out`; * will be set to `NULL` unless `path` is `NULL` * @return `FILE` object for the reading the file; `NULL` if not * found and `path` is `NULL` */ static FILE * open_config_file(const char *path, const char **path_out, char **pathbuf_out) { FILE *f = NULL; size_t i; if (!path) { for (i = 0; !f && i < ELEMSOF(paths); i++) f = try_path(&paths[i], pathbuf_out); if (!f) weprintf(_("No configuration file found.")); *path_out = *pathbuf_out; } else { f = fopen(path, "r"); if (!f) eprintf("fopen %s \"r\":", path); *path_out = path; *pathbuf_out = NULL; } return f; } void config_ini_init(struct config_ini_state *state, const char *path) { struct config_ini_section *section = NULL; struct config_ini_setting *setting; char *line = NULL, *s, *p, *value, *end, *pathbuf; char *next_line = NULL, *name; size_t size = 0; ssize_t len; FILE *f; state->sections = NULL; f = open_config_file(path, &path, &pathbuf); if (!f) return; while ((s = next_line) || (len = getline(&line, &size, f)) >= 0) { if (!s && (s = line, strlen(s) != (size_t)len)) eprintf(_("Config file contains NUL byte.")); s = ltrim(s); next_line = NULL; end = &s[strcspn(s, "\r\n")]; if (*end) { p = end; do { *p++ = '\0'; } while (*p == '\r' || *p == '\n'); if (*p) next_line = p; } rtrim(s, end); switch (*s) { case ';': /* comment line */ case '#': /* comment line */ case '\0': /* blank line */ break; case '[': /* "[%s]", section */ end = strchr(name = &s[1], ']'); if (!end || end[1] || end == name) eprintf(_("Malformed section header in config file.")); *end = '\0'; section = emalloc(sizeof(*section)); section->name = estrdup(name); section->settings = NULL; section->next = state->sections; state->sections = section; break; default: /* "%s = %s", name, value */ value = p = strchr(name = s, '='); if (!value || value == name) eprintf(_("Malformed assignment in config file.")); *value++ = '\0'; if (!section) eprintf(_("Assignment outside section in config file.")); setting = emalloc(sizeof(*setting)); setting->name = estrdup(rtrim(name, p)); setting->value = estrdup(rtrim(ltrim(value), NULL)); setting->next = section->settings; section->settings = setting; break; } } if (ferror(f)) eprintf("getline %s:", path); free(pathbuf); free(line); fclose(f); } void config_ini_free(struct config_ini_state *state) { struct config_ini_section *section, *section_next; struct config_ini_setting *setting, *setting_next; for (section = state->sections; section; section = section_next) { section_next = section->next; for (setting = section->settings; setting; setting = setting_next) { setting_next = setting->next; free(setting->name); free(setting->value); free(setting); } free(section->name); free(section); } } struct config_ini_section * config_ini_get_section(struct config_ini_state *state, const char *name) { struct config_ini_section *section; for (section = state->sections; section; section = section->next) if (!strcasecmp(section->name, name)) return section; return NULL; }