Factor out XDG base directory code
And add warnings to configOpen, since that's the only way to be accurate if a weird error occurs.master
parent
e6c18403e2
commit
99480a42e5
1
Makefile
1
Makefile
|
@ -15,6 +15,7 @@ OBJS += handle.o
|
||||||
OBJS += irc.o
|
OBJS += irc.o
|
||||||
OBJS += ui.o
|
OBJS += ui.o
|
||||||
OBJS += url.o
|
OBJS += url.o
|
||||||
|
OBJS += xdg.o
|
||||||
|
|
||||||
dev: tags all
|
dev: tags all
|
||||||
|
|
||||||
|
|
4
chat.c
4
chat.c
|
@ -154,11 +154,11 @@ int main(int argc, char *argv[]) {
|
||||||
FILE *privFile = NULL;
|
FILE *privFile = NULL;
|
||||||
if (cert) {
|
if (cert) {
|
||||||
certFile = configOpen(cert, "r");
|
certFile = configOpen(cert, "r");
|
||||||
if (!certFile) err(EX_NOINPUT, "%s", cert);
|
if (!certFile) return EX_NOINPUT;
|
||||||
}
|
}
|
||||||
if (priv) {
|
if (priv) {
|
||||||
privFile = configOpen(priv, "r");
|
privFile = configOpen(priv, "r");
|
||||||
if (!privFile) err(EX_NOINPUT, "%s", priv);
|
if (!privFile) return EX_NOINPUT;
|
||||||
}
|
}
|
||||||
ircConfig(insecure, certFile, privFile);
|
ircConfig(insecure, certFile, privFile);
|
||||||
if (certFile) fclose(certFile);
|
if (certFile) fclose(certFile);
|
||||||
|
|
2
chat.h
2
chat.h
|
@ -189,6 +189,8 @@ void urlOpenMatch(size_t id, const char *str);
|
||||||
void urlCopyMatch(size_t id, const char *str);
|
void urlCopyMatch(size_t id, const char *str);
|
||||||
|
|
||||||
FILE *configOpen(const char *path, const char *mode);
|
FILE *configOpen(const char *path, const char *mode);
|
||||||
|
FILE *dataOpen(const char *path, const char *mode);
|
||||||
|
|
||||||
int getopt_config(
|
int getopt_config(
|
||||||
int argc, char *const *argv,
|
int argc, char *const *argv,
|
||||||
const char *optstring, const struct option *longopts, int *longindex
|
const char *optstring, const struct option *longopts, int *longindex
|
||||||
|
|
41
config.c
41
config.c
|
@ -24,42 +24,6 @@
|
||||||
|
|
||||||
#include "chat.h"
|
#include "chat.h"
|
||||||
|
|
||||||
FILE *configOpen(const char *path, const char *mode) {
|
|
||||||
if (path[0] == '/' || path[0] == '.') goto local;
|
|
||||||
|
|
||||||
const char *home = getenv("HOME");
|
|
||||||
const char *configHome = getenv("XDG_CONFIG_HOME");
|
|
||||||
const char *configDirs = getenv("XDG_CONFIG_DIRS");
|
|
||||||
|
|
||||||
char buf[PATH_MAX];
|
|
||||||
if (configHome) {
|
|
||||||
snprintf(buf, sizeof(buf), "%s/" XDG_SUBDIR "/%s", configHome, path);
|
|
||||||
} else {
|
|
||||||
if (!home) goto local;
|
|
||||||
snprintf(buf, sizeof(buf), "%s/.config/" XDG_SUBDIR "/%s", home, path);
|
|
||||||
}
|
|
||||||
FILE *file = fopen(buf, mode);
|
|
||||||
if (file) return file;
|
|
||||||
if (errno != ENOENT) return NULL;
|
|
||||||
|
|
||||||
if (!configDirs) configDirs = "/etc/xdg";
|
|
||||||
while (*configDirs) {
|
|
||||||
size_t len = strcspn(configDirs, ":");
|
|
||||||
snprintf(
|
|
||||||
buf, sizeof(buf), "%.*s/" XDG_SUBDIR "/%s",
|
|
||||||
(int)len, configDirs, path
|
|
||||||
);
|
|
||||||
file = fopen(buf, mode);
|
|
||||||
if (file) return file;
|
|
||||||
if (errno != ENOENT) return NULL;
|
|
||||||
configDirs += len;
|
|
||||||
if (*configDirs) configDirs++;
|
|
||||||
}
|
|
||||||
|
|
||||||
local:
|
|
||||||
return fopen(path, mode);
|
|
||||||
}
|
|
||||||
|
|
||||||
#define WS "\t "
|
#define WS "\t "
|
||||||
|
|
||||||
static const char *path;
|
static const char *path;
|
||||||
|
@ -92,10 +56,7 @@ int getopt_config(
|
||||||
num = 0;
|
num = 0;
|
||||||
path = argv[optind++];
|
path = argv[optind++];
|
||||||
file = configOpen(path, "r");
|
file = configOpen(path, "r");
|
||||||
if (!file) {
|
if (!file) return clean('?');
|
||||||
warn("%s", path);
|
|
||||||
return clean('?');
|
|
||||||
}
|
|
||||||
} else {
|
} else {
|
||||||
return clean(-1);
|
return clean(-1);
|
||||||
}
|
}
|
||||||
|
|
67
ui.c
67
ui.c
|
@ -21,13 +21,11 @@
|
||||||
#include <curses.h>
|
#include <curses.h>
|
||||||
#include <err.h>
|
#include <err.h>
|
||||||
#include <errno.h>
|
#include <errno.h>
|
||||||
#include <limits.h>
|
|
||||||
#include <stdarg.h>
|
#include <stdarg.h>
|
||||||
#include <stdbool.h>
|
#include <stdbool.h>
|
||||||
#include <stdio.h>
|
#include <stdio.h>
|
||||||
#include <stdlib.h>
|
#include <stdlib.h>
|
||||||
#include <string.h>
|
#include <string.h>
|
||||||
#include <sys/stat.h>
|
|
||||||
#include <sysexits.h>
|
#include <sysexits.h>
|
||||||
#include <term.h>
|
#include <term.h>
|
||||||
#include <termios.h>
|
#include <termios.h>
|
||||||
|
@ -858,71 +856,6 @@ void uiRead(void) {
|
||||||
inputUpdate();
|
inputUpdate();
|
||||||
}
|
}
|
||||||
|
|
||||||
static FILE *dataOpen(const char *path, const char *mode) {
|
|
||||||
if (path[0] == '/' || path[0] == '.') goto local;
|
|
||||||
|
|
||||||
const char *home = getenv("HOME");
|
|
||||||
const char *dataHome = getenv("XDG_DATA_HOME");
|
|
||||||
const char *dataDirs = getenv("XDG_DATA_DIRS");
|
|
||||||
|
|
||||||
char homePath[PATH_MAX];
|
|
||||||
if (dataHome) {
|
|
||||||
snprintf(
|
|
||||||
homePath, sizeof(homePath),
|
|
||||||
"%s/" XDG_SUBDIR "/%s", dataHome, path
|
|
||||||
);
|
|
||||||
} else {
|
|
||||||
if (!home) goto local;
|
|
||||||
snprintf(
|
|
||||||
homePath, sizeof(homePath),
|
|
||||||
"%s/.local/share/" XDG_SUBDIR "/%s", home, path
|
|
||||||
);
|
|
||||||
}
|
|
||||||
FILE *file = fopen(homePath, mode);
|
|
||||||
if (file) return file;
|
|
||||||
if (errno != ENOENT) {
|
|
||||||
warn("%s", homePath);
|
|
||||||
return NULL;
|
|
||||||
}
|
|
||||||
|
|
||||||
char buf[PATH_MAX];
|
|
||||||
if (!dataDirs) dataDirs = "/usr/local/share:/usr/share";
|
|
||||||
while (*dataDirs) {
|
|
||||||
size_t len = strcspn(dataDirs, ":");
|
|
||||||
snprintf(
|
|
||||||
buf, sizeof(buf), "%.*s/" XDG_SUBDIR "/%s",
|
|
||||||
(int)len, dataDirs, path
|
|
||||||
);
|
|
||||||
file = fopen(buf, mode);
|
|
||||||
if (file) return file;
|
|
||||||
if (errno != ENOENT) {
|
|
||||||
warn("%s", buf);
|
|
||||||
return NULL;
|
|
||||||
}
|
|
||||||
dataDirs += len;
|
|
||||||
if (*dataDirs) dataDirs++;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (mode[0] != 'r') {
|
|
||||||
char *base = strrchr(homePath, '/');
|
|
||||||
*base = '\0';
|
|
||||||
int error = mkdir(homePath, S_IRWXU);
|
|
||||||
if (error && errno != EEXIST) {
|
|
||||||
warn("%s", homePath);
|
|
||||||
return NULL;
|
|
||||||
}
|
|
||||||
*base = '/';
|
|
||||||
file = fopen(homePath, mode);
|
|
||||||
if (!file) warn("%s", homePath);
|
|
||||||
return file;
|
|
||||||
}
|
|
||||||
|
|
||||||
local:
|
|
||||||
file = fopen(path, mode);
|
|
||||||
if (!file) warn("%s", path);
|
|
||||||
return file;
|
|
||||||
}
|
|
||||||
|
|
||||||
static const size_t Signatures[] = {
|
static const size_t Signatures[] = {
|
||||||
0x6C72696774616301,
|
0x6C72696774616301,
|
||||||
};
|
};
|
||||||
|
|
|
@ -0,0 +1,134 @@
|
||||||
|
/* Copyright (C) 2019, 2020 C. McEnroe <june@causal.agency>
|
||||||
|
*
|
||||||
|
* This program 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.
|
||||||
|
*
|
||||||
|
* This program 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 this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include <err.h>
|
||||||
|
#include <errno.h>
|
||||||
|
#include <limits.h>
|
||||||
|
#include <stdio.h>
|
||||||
|
#include <stdlib.h>
|
||||||
|
#include <string.h>
|
||||||
|
#include <sys/stat.h>
|
||||||
|
|
||||||
|
#include "chat.h"
|
||||||
|
|
||||||
|
FILE *configOpen(const char *path, const char *mode) {
|
||||||
|
if (path[0] == '/' || path[0] == '.') goto local;
|
||||||
|
|
||||||
|
const char *home = getenv("HOME");
|
||||||
|
const char *configHome = getenv("XDG_CONFIG_HOME");
|
||||||
|
const char *configDirs = getenv("XDG_CONFIG_DIRS");
|
||||||
|
|
||||||
|
char buf[PATH_MAX];
|
||||||
|
if (configHome) {
|
||||||
|
snprintf(buf, sizeof(buf), "%s/" XDG_SUBDIR "/%s", configHome, path);
|
||||||
|
} else {
|
||||||
|
if (!home) goto local;
|
||||||
|
snprintf(buf, sizeof(buf), "%s/.config/" XDG_SUBDIR "/%s", home, path);
|
||||||
|
}
|
||||||
|
FILE *file = fopen(buf, mode);
|
||||||
|
if (file) return file;
|
||||||
|
if (errno != ENOENT) {
|
||||||
|
warn("%s", buf);
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!configDirs) configDirs = "/etc/xdg";
|
||||||
|
while (*configDirs) {
|
||||||
|
size_t len = strcspn(configDirs, ":");
|
||||||
|
snprintf(
|
||||||
|
buf, sizeof(buf), "%.*s/" XDG_SUBDIR "/%s",
|
||||||
|
(int)len, configDirs, path
|
||||||
|
);
|
||||||
|
file = fopen(buf, mode);
|
||||||
|
if (file) return file;
|
||||||
|
if (errno != ENOENT) {
|
||||||
|
warn("%s", buf);
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
configDirs += len;
|
||||||
|
if (*configDirs) configDirs++;
|
||||||
|
}
|
||||||
|
|
||||||
|
local:
|
||||||
|
file = fopen(path, mode);
|
||||||
|
if (!file) warn("%s", path);
|
||||||
|
return file;
|
||||||
|
}
|
||||||
|
|
||||||
|
FILE *dataOpen(const char *path, const char *mode) {
|
||||||
|
if (path[0] == '/' || path[0] == '.') goto local;
|
||||||
|
|
||||||
|
const char *home = getenv("HOME");
|
||||||
|
const char *dataHome = getenv("XDG_DATA_HOME");
|
||||||
|
const char *dataDirs = getenv("XDG_DATA_DIRS");
|
||||||
|
|
||||||
|
char homePath[PATH_MAX];
|
||||||
|
if (dataHome) {
|
||||||
|
snprintf(
|
||||||
|
homePath, sizeof(homePath),
|
||||||
|
"%s/" XDG_SUBDIR "/%s", dataHome, path
|
||||||
|
);
|
||||||
|
} else {
|
||||||
|
if (!home) goto local;
|
||||||
|
snprintf(
|
||||||
|
homePath, sizeof(homePath),
|
||||||
|
"%s/.local/share/" XDG_SUBDIR "/%s", home, path
|
||||||
|
);
|
||||||
|
}
|
||||||
|
FILE *file = fopen(homePath, mode);
|
||||||
|
if (file) return file;
|
||||||
|
if (errno != ENOENT) {
|
||||||
|
warn("%s", homePath);
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
char buf[PATH_MAX];
|
||||||
|
if (!dataDirs) dataDirs = "/usr/local/share:/usr/share";
|
||||||
|
while (*dataDirs) {
|
||||||
|
size_t len = strcspn(dataDirs, ":");
|
||||||
|
snprintf(
|
||||||
|
buf, sizeof(buf), "%.*s/" XDG_SUBDIR "/%s",
|
||||||
|
(int)len, dataDirs, path
|
||||||
|
);
|
||||||
|
file = fopen(buf, mode);
|
||||||
|
if (file) return file;
|
||||||
|
if (errno != ENOENT) {
|
||||||
|
warn("%s", buf);
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
dataDirs += len;
|
||||||
|
if (*dataDirs) dataDirs++;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (mode[0] != 'r') {
|
||||||
|
char *base = strrchr(homePath, '/');
|
||||||
|
*base = '\0';
|
||||||
|
int error = mkdir(homePath, S_IRWXU);
|
||||||
|
if (error && errno != EEXIST) {
|
||||||
|
warn("%s", homePath);
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
*base = '/';
|
||||||
|
file = fopen(homePath, mode);
|
||||||
|
if (!file) warn("%s", homePath);
|
||||||
|
return file;
|
||||||
|
}
|
||||||
|
|
||||||
|
local:
|
||||||
|
file = fopen(path, mode);
|
||||||
|
if (!file) warn("%s", path);
|
||||||
|
return file;
|
||||||
|
}
|
Loading…
Reference in New Issue