2015-03-28 19:30:35 +01:00
|
|
|
/*
|
|
|
|
* vim:ts=4:sw=4:expandtab
|
|
|
|
*
|
|
|
|
* i3 - an improved dynamic tiling window manager
|
2015-04-04 02:17:56 +02:00
|
|
|
* © 2009 Michael Stapelberg and contributors (see also: LICENSE)
|
2015-03-28 19:30:35 +01:00
|
|
|
*
|
|
|
|
*/
|
|
|
|
#include "libi3.h"
|
2016-10-11 09:13:35 +02:00
|
|
|
|
2015-03-28 19:30:35 +01:00
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <sys/stat.h>
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Checks if the given path exists by calling stat().
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
static bool path_exists(const char *path) {
|
|
|
|
struct stat buf;
|
|
|
|
return (stat(path, &buf) == 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
2018-07-10 04:04:34 +02:00
|
|
|
* Get the path of the first configuration file found. If override_configpath is
|
|
|
|
* specified, that path is returned and saved for further calls. Otherwise,
|
|
|
|
* checks the home directory first, then the system directory, always taking
|
|
|
|
* into account the XDG Base Directory Specification ($XDG_CONFIG_HOME,
|
|
|
|
* $XDG_CONFIG_DIRS).
|
2015-03-28 19:30:35 +01:00
|
|
|
*
|
|
|
|
*/
|
|
|
|
char *get_config_path(const char *override_configpath, bool use_system_paths) {
|
|
|
|
char *xdg_config_home, *xdg_config_dirs, *config_path;
|
|
|
|
|
|
|
|
static const char *saved_configpath = NULL;
|
|
|
|
|
|
|
|
if (override_configpath != NULL) {
|
|
|
|
saved_configpath = override_configpath;
|
|
|
|
return sstrdup(saved_configpath);
|
|
|
|
}
|
|
|
|
|
2018-07-10 04:04:34 +02:00
|
|
|
if (saved_configpath != NULL) {
|
2015-03-28 19:30:35 +01:00
|
|
|
return sstrdup(saved_configpath);
|
2018-07-10 04:04:34 +02:00
|
|
|
}
|
2015-03-28 19:30:35 +01:00
|
|
|
|
2018-07-10 04:04:34 +02:00
|
|
|
/* 1: check for $XDG_CONFIG_HOME/i3/config */
|
|
|
|
if ((xdg_config_home = getenv("XDG_CONFIG_HOME")) == NULL) {
|
2015-03-28 19:30:35 +01:00
|
|
|
xdg_config_home = "~/.config";
|
2018-07-10 04:04:34 +02:00
|
|
|
}
|
2015-03-28 19:30:35 +01:00
|
|
|
|
|
|
|
xdg_config_home = resolve_tilde(xdg_config_home);
|
|
|
|
sasprintf(&config_path, "%s/i3/config", xdg_config_home);
|
|
|
|
free(xdg_config_home);
|
|
|
|
|
2018-07-10 04:04:34 +02:00
|
|
|
if (path_exists(config_path)) {
|
2015-03-28 19:30:35 +01:00
|
|
|
return config_path;
|
2018-07-10 04:04:34 +02:00
|
|
|
}
|
|
|
|
free(config_path);
|
|
|
|
|
|
|
|
/* 2: check the traditional path under the home directory */
|
|
|
|
config_path = resolve_tilde("~/.i3/config");
|
|
|
|
if (path_exists(config_path)) {
|
|
|
|
return config_path;
|
|
|
|
}
|
2015-03-28 19:30:35 +01:00
|
|
|
free(config_path);
|
|
|
|
|
|
|
|
/* The below paths are considered system-level, and can be skipped if the
|
|
|
|
* caller only wants user-level configs. */
|
2018-07-10 04:04:34 +02:00
|
|
|
if (!use_system_paths) {
|
2015-03-28 19:30:35 +01:00
|
|
|
return NULL;
|
2018-07-10 04:04:34 +02:00
|
|
|
}
|
2015-03-28 19:30:35 +01:00
|
|
|
|
2018-07-10 04:04:34 +02:00
|
|
|
/* 3: check for $XDG_CONFIG_DIRS/i3/config */
|
|
|
|
if ((xdg_config_dirs = getenv("XDG_CONFIG_DIRS")) == NULL) {
|
2016-11-09 22:34:39 +01:00
|
|
|
xdg_config_dirs = SYSCONFDIR "/xdg";
|
2018-07-10 04:04:34 +02:00
|
|
|
}
|
2015-03-28 19:30:35 +01:00
|
|
|
|
|
|
|
char *buf = sstrdup(xdg_config_dirs);
|
|
|
|
char *tok = strtok(buf, ":");
|
|
|
|
while (tok != NULL) {
|
|
|
|
tok = resolve_tilde(tok);
|
|
|
|
sasprintf(&config_path, "%s/i3/config", tok);
|
|
|
|
free(tok);
|
|
|
|
if (path_exists(config_path)) {
|
|
|
|
free(buf);
|
|
|
|
return config_path;
|
|
|
|
}
|
|
|
|
free(config_path);
|
|
|
|
tok = strtok(NULL, ":");
|
|
|
|
}
|
|
|
|
free(buf);
|
|
|
|
|
2018-07-10 04:04:34 +02:00
|
|
|
/* 4: check the traditional path under /etc */
|
|
|
|
config_path = SYSCONFDIR "/i3/config";
|
|
|
|
if (path_exists(config_path)) {
|
|
|
|
return sstrdup(config_path);
|
|
|
|
}
|
|
|
|
|
2015-03-28 19:30:35 +01:00
|
|
|
return NULL;
|
|
|
|
}
|