diff options
-rw-r--r-- | usr.sbin/unwindctl/Makefile | 17 | ||||
-rw-r--r-- | usr.sbin/unwindctl/parser.c | 175 | ||||
-rw-r--r-- | usr.sbin/unwindctl/parser.h | 36 | ||||
-rw-r--r-- | usr.sbin/unwindctl/unwindctl.8 | 81 | ||||
-rw-r--r-- | usr.sbin/unwindctl/unwindctl.c | 276 |
5 files changed, 585 insertions, 0 deletions
diff --git a/usr.sbin/unwindctl/Makefile b/usr.sbin/unwindctl/Makefile new file mode 100644 index 00000000000..790e451757b --- /dev/null +++ b/usr.sbin/unwindctl/Makefile @@ -0,0 +1,17 @@ +# $OpenBSD: Makefile,v 1.1 2019/01/23 13:12:19 florian Exp $ + +PROG= unwindctl +SRCS= unwindctl.c parser.c + +MAN= unwindctl.8 + +CFLAGS+= -Wall +CFLAGS+= -Wstrict-prototypes -Wmissing-prototypes +CFLAGS+= -Wmissing-declarations +CFLAGS+= -Wshadow -Wpointer-arith -Wcast-qual +CFLAGS+= -Wsign-compare +CFLAGS+= -I${.CURDIR} -I${.CURDIR}/../../sbin/unwind +LDADD= -lutil +DPADD= ${LIBUTIL} + +.include <bsd.prog.mk> diff --git a/usr.sbin/unwindctl/parser.c b/usr.sbin/unwindctl/parser.c new file mode 100644 index 00000000000..3f8fac4bb27 --- /dev/null +++ b/usr.sbin/unwindctl/parser.c @@ -0,0 +1,175 @@ +/* $OpenBSD: parser.c,v 1.1 2019/01/23 13:12:19 florian Exp $ */ + +/* + * Copyright (c) 2004 Esben Norby <norby@openbsd.org> + * Copyright (c) 2003, 2004 Henning Brauer <henning@openbsd.org> + * + * 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 <sys/types.h> +#include <sys/queue.h> +#include <sys/socket.h> +#include <netinet/in.h> +#include <arpa/inet.h> +#include <net/if.h> +#include <err.h> +#include <errno.h> +#include <event.h> +#include <imsg.h> +#include <limits.h> +#include <stdio.h> +#include <stdlib.h> +#include <string.h> + +#include "unwind.h" +#include "parser.h" + +enum token_type { + NOTOKEN, + ENDTOKEN, + KEYWORD +}; + +struct token { + enum token_type type; + const char *keyword; + int value; + const struct token *next; +}; + +static const struct token t_main[]; +static const struct token t_log[]; +static const struct token t_status[]; + +static const struct token t_main[] = { + {KEYWORD, "reload", RELOAD, NULL}, + {KEYWORD, "status", STATUS, t_status}, + {KEYWORD, "log", NONE, t_log}, + {ENDTOKEN, "", NONE, NULL} +}; + +static const struct token t_log[] = { + {KEYWORD, "debug", LOG_DEBUG, NULL}, + {KEYWORD, "verbose", LOG_VERBOSE, NULL}, + {KEYWORD, "brief", LOG_BRIEF, NULL}, + {ENDTOKEN, "", NONE, NULL} +}; + +static const struct token t_status[] = { + {NOTOKEN, "", NONE, NULL}, + {KEYWORD, "recursor", STATUS_RECURSOR, NULL}, + {KEYWORD, "dhcp", STATUS_DHCP, NULL}, + {KEYWORD, "static", STATUS_STATIC, NULL}, + {ENDTOKEN, "", STATUS, NULL} +}; + +static const struct token *match_token(const char *, const struct token *, + struct parse_result *); +static void show_valid_args(const struct token *); + +struct parse_result * +parse(int argc, char *argv[]) +{ + static struct parse_result res; + const struct token *table = t_main; + const struct token *match; + + memset(&res, 0, sizeof(res)); + + while (argc >= 0) { + if ((match = match_token(argv[0], table, &res)) == NULL) { + fprintf(stderr, "valid commands/args:\n"); + show_valid_args(table); + return (NULL); + } + + argc--; + argv++; + + if (match->type == NOTOKEN || match->next == NULL) + break; + + table = match->next; + } + + if (argc > 0) { + fprintf(stderr, "superfluous argument: %s\n", argv[0]); + return (NULL); + } + + return (&res); +} + +static const struct token * +match_token(const char *word, const struct token *table, + struct parse_result *res) +{ + u_int i, match; + const struct token *t = NULL; + + match = 0; + + for (i = 0; table[i].type != ENDTOKEN; i++) { + switch (table[i].type) { + case NOTOKEN: + if (word == NULL || strlen(word) == 0) { + match++; + t = &table[i]; + } + break; + case KEYWORD: + if (word != NULL && strncmp(word, table[i].keyword, + strlen(word)) == 0) { + match++; + t = &table[i]; + if (t->value) + res->action = t->value; + } + break; + case ENDTOKEN: + break; + } + } + + if (match != 1) { + if (word == NULL) + fprintf(stderr, "missing argument:\n"); + else if (match > 1) + fprintf(stderr, "ambiguous argument: %s\n", word); + else if (match < 1) + fprintf(stderr, "unknown argument: %s\n", word); + return (NULL); + } + + return (t); +} + +static void +show_valid_args(const struct token *table) +{ + int i; + + for (i = 0; table[i].type != ENDTOKEN; i++) { + switch (table[i].type) { + case NOTOKEN: + fprintf(stderr, " <cr>\n"); + break; + case KEYWORD: + fprintf(stderr, " %s\n", table[i].keyword); + break; + case ENDTOKEN: + break; + } + } +} diff --git a/usr.sbin/unwindctl/parser.h b/usr.sbin/unwindctl/parser.h new file mode 100644 index 00000000000..65efcb7e10b --- /dev/null +++ b/usr.sbin/unwindctl/parser.h @@ -0,0 +1,36 @@ +/* $OpenBSD: parser.h,v 1.1 2019/01/23 13:12:19 florian Exp $ */ + +/* + * Copyright (c) 2004 Esben Norby <norby@openbsd.org> + * Copyright (c) 2003, 2004 Henning Brauer <henning@openbsd.org> + * + * 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. + */ + +enum actions { + NONE, + LOG_DEBUG, + LOG_VERBOSE, + LOG_BRIEF, + RELOAD, + STATUS, + STATUS_RECURSOR, + STATUS_DHCP, + STATUS_STATIC +}; + +struct parse_result { + enum actions action; +}; + +struct parse_result *parse(int, char *[]); diff --git a/usr.sbin/unwindctl/unwindctl.8 b/usr.sbin/unwindctl/unwindctl.8 new file mode 100644 index 00000000000..f8dd5a47b2e --- /dev/null +++ b/usr.sbin/unwindctl/unwindctl.8 @@ -0,0 +1,81 @@ +.\" $OpenBSD: unwindctl.8,v 1.1 2019/01/23 13:12:19 florian Exp $ +.\" +.\" Copyright (c) 2004, 2005 Esben Norby <norby@openbsd.org> +.\" +.\" 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. +.\" +.Dd $Mdocdate: January 23 2019 $ +.Dt UNWINDCTL 8 +.Os +.Sh NAME +.Nm unwindctl +.Nd control the unwind daemon +.Sh SYNOPSIS +.Nm +.Op Fl s Ar socket +.Ar command +.Op Ar argument ... +.Sh DESCRIPTION +The +.Nm +program controls the +.Xr unwind 8 +daemon. +.Pp +The following options are available: +.Bl -tag -width Ds +.It Fl s Ar socket +Use +.Ar socket +instead of the default +.Pa /var/run/unwind.sock +to communicate with +.Xr unwind 8 . +.El +.Pp +The following commands are available: +.Bl -tag -width Ds +.It Cm log brief +Disable verbose logging. +.It Cm log verbose +Enable verbose logging. +.It Cm log debug +Enable very noisy debug logging. +.It Cm reload +Reload the configuration file. +.It Cm status Op Cm recursor Cm dhcp Cm static +Show a status summary. +If one of +.Cm recursor , +.Cm dhcp +or +.Cm static +is given more detailed information is given. +Including reasons why DNSSEC validation might be failing and a query time +histogram. +.El +.Sh FILES +.Bl -tag -width "/var/run/unwind.sockXX" -compact +.It Pa /var/run/unwind.sock +.Ux Ns -domain +socket used for communication with +.Xr unwind 8 . +.El +.Sh SEE ALSO +.Xr unwind.conf 5 , +.Xr unwind 8 +.Sh HISTORY +The +.Nm +program first appeared in +.Ox 6.5 . diff --git a/usr.sbin/unwindctl/unwindctl.c b/usr.sbin/unwindctl/unwindctl.c new file mode 100644 index 00000000000..4cd62d4908b --- /dev/null +++ b/usr.sbin/unwindctl/unwindctl.c @@ -0,0 +1,276 @@ +/* $OpenBSD: unwindctl.c,v 1.1 2019/01/23 13:12:19 florian Exp $ */ + +/* + * Copyright (c) 2005 Claudio Jeker <claudio@openbsd.org> + * Copyright (c) 2004, 2005 Esben Norby <norby@openbsd.org> + * Copyright (c) 2003 Henning Brauer <henning@openbsd.org> + * + * 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 <sys/types.h> +#include <sys/queue.h> +#include <sys/socket.h> +#include <sys/un.h> +#include <netinet/in.h> +#include <arpa/inet.h> +#include <net/if.h> +#include <net/if_media.h> +#include <net/if_types.h> + +#include <err.h> +#include <errno.h> +#include <event.h> +#include <imsg.h> +#include <stdio.h> +#include <stdlib.h> +#include <string.h> +#include <unistd.h> + +#include "unwind.h" +#include "frontend.h" +#include "resolver.h" +#include "parser.h" + +__dead void usage(void); +int show_status_msg(struct imsg *); +void print_indented_str(char *); +void print_histogram(void*, size_t len); + +struct imsgbuf *ibuf; + +__dead void +usage(void) +{ + extern char *__progname; + + fprintf(stderr, "usage: %s [-s socket] command [argument ...]\n", + __progname); + exit(1); +} + +int +main(int argc, char *argv[]) +{ + struct sockaddr_un sun; + struct parse_result *res; + struct imsg imsg; + int ctl_sock; + int done = 0; + int n, verbose = 0; + int ch; + int type; + char *sockname; + + sockname = UNWIND_SOCKET; + while ((ch = getopt(argc, argv, "s:")) != -1) { + switch (ch) { + case 's': + sockname = optarg; + break; + default: + usage(); + } + } + argc -= optind; + argv += optind; + + /* Parse command line. */ + if ((res = parse(argc, argv)) == NULL) + exit(1); + + /* Connect to control socket. */ + if ((ctl_sock = socket(AF_UNIX, SOCK_STREAM, 0)) == -1) + err(1, "socket"); + + memset(&sun, 0, sizeof(sun)); + sun.sun_family = AF_UNIX; + + strlcpy(sun.sun_path, sockname, sizeof(sun.sun_path)); + if (connect(ctl_sock, (struct sockaddr *)&sun, sizeof(sun)) == -1) + err(1, "connect: %s", sockname); + + if (pledge("stdio", NULL) == -1) + err(1, "pledge"); + + if ((ibuf = malloc(sizeof(struct imsgbuf))) == NULL) + err(1, NULL); + imsg_init(ibuf, ctl_sock); + done = 0; + + /* Process user request. */ + switch (res->action) { + case LOG_DEBUG: + verbose |= OPT_VERBOSE2; + /* FALLTHROUGH */ + case LOG_VERBOSE: + verbose |= OPT_VERBOSE; + /* FALLTHROUGH */ + case LOG_BRIEF: + imsg_compose(ibuf, IMSG_CTL_LOG_VERBOSE, 0, 0, -1, + &verbose, sizeof(verbose)); + printf("logging request sent.\n"); + done = 1; + break; + case RELOAD: + imsg_compose(ibuf, IMSG_CTL_RELOAD, 0, 0, -1, NULL, 0); + printf("reload request sent.\n"); + done = 1; + break; + case STATUS_RECURSOR: + type = RECURSOR; + imsg_compose(ibuf, IMSG_CTL_STATUS, 0, 0, -1, &type, + sizeof(type)); + break; + case STATUS_DHCP: + type = FORWARDER; + imsg_compose(ibuf, IMSG_CTL_STATUS, 0, 0, -1, &type, + sizeof(type)); + break; + case STATUS_STATIC: + type = STATIC_FORWARDER; + imsg_compose(ibuf, IMSG_CTL_STATUS, 0, 0, -1, &type, + sizeof(type)); + break; + case STATUS: + type = RESOLVER_NONE; + imsg_compose(ibuf, IMSG_CTL_STATUS, 0, 0, -1, &type, + sizeof(type)); + break; + default: + usage(); + } + + while (ibuf->w.queued) + if (msgbuf_write(&ibuf->w) <= 0 && errno != EAGAIN) + err(1, "write error"); + + while (!done) { + if ((n = imsg_read(ibuf)) == -1 && errno != EAGAIN) + errx(1, "imsg_read error"); + if (n == 0) + errx(1, "pipe closed"); + + while (!done) { + if ((n = imsg_get(ibuf, &imsg)) == -1) + errx(1, "imsg_get error"); + if (n == 0) + break; + + switch (res->action) { + case STATUS: + case STATUS_RECURSOR: + case STATUS_DHCP: + case STATUS_STATIC: + done = show_status_msg(&imsg); + break; + default: + break; + } + imsg_free(&imsg); + } + } + close(ctl_sock); + free(ibuf); + + return (0); +} + +int +show_status_msg(struct imsg *imsg) +{ + static int header; + struct ctl_resolver_info *cri; + + if (!header++) + printf("%8s %16s %s\n", "selected", "type", "status"); + + switch (imsg->hdr.type) { + case IMSG_CTL_RESOLVER_INFO: + cri = imsg->data; + printf("%8s %16s %s\n", cri->selected ? "*" : " ", + unwind_resolver_type_str[cri->type], + unwind_resolver_state_str[cri->state]); + break; + case IMSG_CTL_RESOLVER_WHY_BOGUS: + /* make sure this is a string */ + ((char *)imsg->data)[imsg->hdr.len - IMSG_HEADER_SIZE -1] = + '\0'; + printf("\nReason for not validating:\n"); + print_indented_str(imsg->data); + break; + case IMSG_CTL_RESOLVER_HISTOGRAM: + print_histogram(imsg->data, imsg->hdr.len - IMSG_HEADER_SIZE); + break; + case IMSG_CTL_END: + return (1); + default: + break; + } + + return (0); +} + +void +print_indented_str(char * str) +{ + int i; + char *cur; + + if (strlen(str) <= 72) { + printf("\t%s\n", str); + return; + } + + for (i = 71; i >= 0; i--) + if (str[i] == ' ') + break; + + if (i < 0) + cur = strchr(str, ' '); + else + cur = &str[i]; + + + if (cur == NULL) + printf("\t%s\n", str); + else { + *cur = '\0'; + printf("\t%s\n", str); + print_indented_str(cur + 1); + } +} + +void +print_histogram(void* data, size_t len) +{ + int64_t histogram[nitems(histogram_limits)]; + size_t i; + char buf[10]; + + if (len != sizeof(histogram)) + errx(1, "invalid histogram size"); + + printf("\n%40s\n", "histogram[ms]"); + + memcpy(histogram, data, len); + + for(i = 1; i < nitems(histogram_limits) - 1; i++) { + snprintf(buf, sizeof(buf), "<%lld", histogram_limits[i]); + printf("%6s", buf); + } + printf("%6s\n", ">"); + for(i = 1; i < nitems(histogram); i++) + printf("%6lld", histogram[i]); + printf("\n"); +} |