forked from aniani/gmnisrv
Initial config parser
This commit is contained in:
parent
ccae8ffd28
commit
58500c8e53
2
Makefile
2
Makefile
@ -33,6 +33,6 @@ distclean: clean
|
|||||||
|
|
||||||
install: all
|
install: all
|
||||||
mkdir -p $(BINDIR)
|
mkdir -p $(BINDIR)
|
||||||
install -Dm755 gmnisrv $(BINDIR)/gmnisrv
|
install -Dm755 gmnisrv $(DESTDIR)$(BINDIR)/gmnisrv
|
||||||
|
|
||||||
.PHONY: clean distclean docs install
|
.PHONY: clean distclean docs install
|
||||||
|
13
config.ini
Normal file
13
config.ini
Normal file
@ -0,0 +1,13 @@
|
|||||||
|
# Space-separated list of hosts
|
||||||
|
listen=0.0.0.0:1965 [::]:1965
|
||||||
|
|
||||||
|
[:tls]
|
||||||
|
# Path to store certificates on disk
|
||||||
|
store=/var/lib/gemini/certs
|
||||||
|
|
||||||
|
# Details for new certificates
|
||||||
|
organization=gmnisrv user
|
||||||
|
email=jsmith@example.org
|
||||||
|
|
||||||
|
[localhost]
|
||||||
|
root=/var/www
|
24
config.sh
24
config.sh
@ -13,6 +13,19 @@ do
|
|||||||
case "$arg" in
|
case "$arg" in
|
||||||
--prefix=*)
|
--prefix=*)
|
||||||
PREFIX=${arg#*=}
|
PREFIX=${arg#*=}
|
||||||
|
if [ "$PREFIX" = "/usr" ]
|
||||||
|
then
|
||||||
|
SYSCONFDIR=/etc
|
||||||
|
fi
|
||||||
|
;;
|
||||||
|
--bindir=*)
|
||||||
|
BINDIR=${arg#*=}
|
||||||
|
;;
|
||||||
|
--sysconfdir=*)
|
||||||
|
SYSCONFDIR=${arg#*=}
|
||||||
|
;;
|
||||||
|
--mandir=*)
|
||||||
|
MANDIR=${arg#*=}
|
||||||
;;
|
;;
|
||||||
esac
|
esac
|
||||||
done
|
done
|
||||||
@ -123,15 +136,18 @@ run_configure() {
|
|||||||
LIBS=$LIBS
|
LIBS=$LIBS
|
||||||
PREFIX=${PREFIX:-/usr/local}
|
PREFIX=${PREFIX:-/usr/local}
|
||||||
OUTDIR=${outdir}
|
OUTDIR=${outdir}
|
||||||
_INSTDIR=\$(DESTDIR)\$(PREFIX)
|
BINDIR?=${BINDIR:-\$(PREFIX)/bin}
|
||||||
BINDIR?=${BINDIR:-\$(_INSTDIR)/bin}
|
SYSCONFDIR?=${SYSCONFDIR:-\$(PREFIX)/etc}
|
||||||
LIBDIR?=${LIBDIR:-\$(_INSTDIR)/lib}
|
LIBDIR?=${LIBDIR:-\$(PREFIX)/lib}
|
||||||
MANDIR?=${MANDIR:-\$(_INSTDIR)/share/man}
|
MANDIR?=${MANDIR:-\$(PREFIX)/share/man}
|
||||||
|
VARLIBDIR?=${MANDIR:-\$(PREFIX)/var/lib}
|
||||||
CACHE=\$(OUTDIR)/cache
|
CACHE=\$(OUTDIR)/cache
|
||||||
CFLAGS=${CFLAGS}
|
CFLAGS=${CFLAGS}
|
||||||
CFLAGS+=-Iinclude -I\$(OUTDIR)
|
CFLAGS+=-Iinclude -I\$(OUTDIR)
|
||||||
CFLAGS+=-DPREFIX='"\$(PREFIX)"'
|
CFLAGS+=-DPREFIX='"\$(PREFIX)"'
|
||||||
CFLAGS+=-DLIBDIR='"\$(LIBDIR)"'
|
CFLAGS+=-DLIBDIR='"\$(LIBDIR)"'
|
||||||
|
CFLAGS+=-DVARLIBDIR='"\$(VARLIBDIR)"'
|
||||||
|
CFLAGS+=-DSYSCONFDIR='"\$(SYSCONFDIR)"'
|
||||||
|
|
||||||
all: ${all}
|
all: ${all}
|
||||||
EOF
|
EOF
|
||||||
|
1
configure
vendored
1
configure
vendored
@ -4,6 +4,7 @@ eval ". $srcdir/config.sh"
|
|||||||
|
|
||||||
gmnisrv() {
|
gmnisrv() {
|
||||||
genrules gmnisrv \
|
genrules gmnisrv \
|
||||||
|
src/config.c \
|
||||||
src/escape.c \
|
src/escape.c \
|
||||||
src/ini.c \
|
src/ini.c \
|
||||||
src/main.c \
|
src/main.c \
|
||||||
|
34
include/config.h
Normal file
34
include/config.h
Normal file
@ -0,0 +1,34 @@
|
|||||||
|
#ifndef GMNISRV_CONFIG
|
||||||
|
#define GMNISRV_CONFIG
|
||||||
|
#include <arpa/inet.h>
|
||||||
|
|
||||||
|
struct gmnisrv_tls {
|
||||||
|
char *store;
|
||||||
|
char *organization;
|
||||||
|
char *email;
|
||||||
|
};
|
||||||
|
|
||||||
|
struct gmnisrv_host {
|
||||||
|
char *hostname;
|
||||||
|
char *root;
|
||||||
|
struct gmnisrv_host *next;
|
||||||
|
};
|
||||||
|
|
||||||
|
struct gmnisrv_bind {
|
||||||
|
int family;
|
||||||
|
char addr[sizeof(struct in6_addr)];
|
||||||
|
struct gmnisrv_bind *next;
|
||||||
|
};
|
||||||
|
|
||||||
|
struct gmnisrv_config {
|
||||||
|
struct gmnisrv_tls tls;
|
||||||
|
struct gmnisrv_host *hosts;
|
||||||
|
struct gmnisrv_bind *binds;
|
||||||
|
};
|
||||||
|
|
||||||
|
int load_config(struct gmnisrv_config *conf, const char *path);
|
||||||
|
|
||||||
|
struct gmnisrv_host *gmnisrv_config_get_host(
|
||||||
|
struct gmnisrv_config *conf, const char *hostname);
|
||||||
|
|
||||||
|
#endif
|
141
src/config.c
Normal file
141
src/config.c
Normal file
@ -0,0 +1,141 @@
|
|||||||
|
#include <errno.h>
|
||||||
|
#include <stdio.h>
|
||||||
|
#include <stdlib.h>
|
||||||
|
#include <string.h>
|
||||||
|
#include "config.h"
|
||||||
|
#include "ini.h"
|
||||||
|
|
||||||
|
struct gmnisrv_host *
|
||||||
|
gmnisrv_config_get_host(struct gmnisrv_config *conf, const char *hostname)
|
||||||
|
{
|
||||||
|
struct gmnisrv_host *host = conf->hosts;
|
||||||
|
while (host) {
|
||||||
|
if (strcmp(host->hostname, hostname) == 0) {
|
||||||
|
return host;
|
||||||
|
}
|
||||||
|
host = host->next;
|
||||||
|
}
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int
|
||||||
|
parse_listen(struct gmnisrv_config *conf, const char *value)
|
||||||
|
{
|
||||||
|
// TODO
|
||||||
|
(void)conf;
|
||||||
|
(void)value;
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int
|
||||||
|
conf_ini_handler(void *user, const char *section,
|
||||||
|
const char *name, const char *value)
|
||||||
|
{
|
||||||
|
struct gmnisrv_config *conf = (struct gmnisrv_config *)user;
|
||||||
|
struct {
|
||||||
|
char *section;
|
||||||
|
char *name;
|
||||||
|
char **value;
|
||||||
|
} strvars[] = {
|
||||||
|
{ ":tls", "store", &conf->tls.store },
|
||||||
|
{ ":tls", "organization", &conf->tls.organization },
|
||||||
|
{ ":tls", "email", &conf->tls.email },
|
||||||
|
};
|
||||||
|
struct {
|
||||||
|
char *section;
|
||||||
|
char *name;
|
||||||
|
int (*fn)(struct gmnisrv_config *conf, const char *value);
|
||||||
|
} fnvars[] = {
|
||||||
|
{ "", "listen", &parse_listen, }
|
||||||
|
};
|
||||||
|
|
||||||
|
for (size_t i = 0; i < sizeof(strvars) / sizeof(strvars[0]); ++i) {
|
||||||
|
if (strcmp(strvars[i].section, section) != 0) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
if (strcmp(strvars[i].name, name) != 0) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
*strvars[i].value = strdup(value);
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
for (size_t i = 0; i < sizeof(fnvars) / sizeof(fnvars[0]); ++i) {
|
||||||
|
if (strcmp(fnvars[i].section, section) != 0) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
if (strcmp(fnvars[i].name, name) != 0) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
return fnvars[i].fn(conf, value);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (section[0] == '\0' || section[0] == ':') {
|
||||||
|
fprintf(stderr, "Unknown config option [%s]%s\n", section, name);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
struct gmnisrv_host *host = gmnisrv_config_get_host(conf, section);
|
||||||
|
if (!host) {
|
||||||
|
host = calloc(1, sizeof(struct gmnisrv_host));
|
||||||
|
host->hostname = strdup(section);
|
||||||
|
host->next = conf->hosts;
|
||||||
|
conf->hosts = host;
|
||||||
|
}
|
||||||
|
|
||||||
|
struct {
|
||||||
|
char *name;
|
||||||
|
char **value;
|
||||||
|
} host_strvars[] = {
|
||||||
|
{ "root", &host->root },
|
||||||
|
};
|
||||||
|
|
||||||
|
for (size_t i = 0; i < sizeof(host_strvars) / sizeof(host_strvars[0]); ++i) {
|
||||||
|
if (strcmp(host_strvars[i].name, name) != 0) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
*host_strvars[i].value = strdup(value);
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
fprintf(stderr, "Unknown config option [%s]%s\n", section, name);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int
|
||||||
|
validate_config(struct gmnisrv_config *conf)
|
||||||
|
{
|
||||||
|
if (!conf->binds) {
|
||||||
|
fprintf(stderr, "Error: config missing listen directive\n");
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
if (!conf->hosts) {
|
||||||
|
fprintf(stderr, "Error: config defines no hosts\n");
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
if (!conf->tls.store) {
|
||||||
|
fprintf(stderr, "Error: config defines no certificate store\n");
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
int
|
||||||
|
load_config(struct gmnisrv_config *conf, const char *path)
|
||||||
|
{
|
||||||
|
FILE *f = fopen(path, "r");
|
||||||
|
if (!f) {
|
||||||
|
fprintf(stderr, "Error opening %s: %s\n",
|
||||||
|
path, strerror(errno));
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
int n = ini_parse_file(f, conf_ini_handler, conf);
|
||||||
|
fclose(f);
|
||||||
|
if (n != 0) {
|
||||||
|
fprintf(stderr, "at %s:%d\n", path, n);
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
return validate_config(conf);
|
||||||
|
}
|
@ -20,8 +20,8 @@ https://github.com/benhoyt/inih
|
|||||||
#include <stdlib.h>
|
#include <stdlib.h>
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#define MAX_SECTION 50
|
#define MAX_SECTION 512
|
||||||
#define MAX_NAME 50
|
#define MAX_NAME 512
|
||||||
|
|
||||||
/* Strip whitespace chars off end of given string, in place. Return s. */
|
/* Strip whitespace chars off end of given string, in place. Return s. */
|
||||||
static char* rstrip(char* s)
|
static char* rstrip(char* s)
|
||||||
|
39
src/main.c
39
src/main.c
@ -1,10 +1,43 @@
|
|||||||
|
#include <getopt.h>
|
||||||
#include <stdio.h>
|
#include <stdio.h>
|
||||||
|
#include "config.h"
|
||||||
|
|
||||||
|
static void
|
||||||
|
usage(const char *argv_0)
|
||||||
|
{
|
||||||
|
fprintf(stderr, "Usage: %s [-C path]\n", argv_0);
|
||||||
|
}
|
||||||
|
|
||||||
int
|
int
|
||||||
main(int argc, char **argv)
|
main(int argc, char **argv)
|
||||||
{
|
{
|
||||||
printf("Hello world!\n");
|
struct gmnisrv_config conf = {0};
|
||||||
(void)argc;
|
|
||||||
(void)argv;
|
char *confpath = SYSCONFDIR "/gmnisrv.ini";
|
||||||
|
int c;
|
||||||
|
while ((c = getopt(argc, argv, "C:h")) != -1) {
|
||||||
|
switch (c) {
|
||||||
|
case 'C':
|
||||||
|
confpath = optarg;
|
||||||
|
break;
|
||||||
|
case 'h':
|
||||||
|
usage(argv[0]);
|
||||||
|
return 0;
|
||||||
|
default:
|
||||||
|
fprintf(stderr, "Unknown flag %c\n", c);
|
||||||
|
usage(argv[0]);
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (optind < argc) {
|
||||||
|
usage(argv[0]);
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
int r = load_config(&conf, confpath);
|
||||||
|
if (r != 0) {
|
||||||
|
return r;
|
||||||
|
}
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user