Add command show

Signed-off-by: Adrien Gallouët <adrien@gallouet.fr>
This commit is contained in:
Adrien Gallouët
2018-02-12 22:58:57 +00:00
parent 7548ba2c27
commit 3c2423a38b
8 changed files with 219 additions and 60 deletions

View File

@@ -199,8 +199,10 @@ gt_bind(int argc, char **argv)
return 1;
}
if (!chacha && mud_set_aes(mud))
if (!chacha && mud_set_aes(mud)) {
gt_log("AES is not available\n");
chacha = 1;
}
if (timeout && mud_set_send_timeout(mud, timeout)) {
perror("timeout");
@@ -234,10 +236,10 @@ gt_bind(int argc, char **argv)
gt_setup_mtu(mud, tun_name, &mtu);
int ctl_fd = ctl_init("/run/" PACKAGE_NAME, tun_name);
int ctl_fd = ctl_create("/run/" PACKAGE_NAME, tun_name);
if (ctl_fd == -1) {
perror("ctl_init");
perror("ctl_create");
return 1;
}
@@ -313,7 +315,21 @@ gt_bind(int argc, char **argv)
perror("mud_del_path");
}
break;
case CTL_PING:
case CTL_STATUS:
gt_log("[ctl status]\n");
reply = (struct ctl_msg){
.type = CTL_STATUS_REPLY,
.status = {
.mtu = mtu,
.mtu_auto = (icmp_fd != -1),
.chacha = chacha,
.port = port,
.bind_port = bind_port,
.ipv4 = ipv4,
.ipv6 = ipv6,
},
};
str_cpy(reply.status.addr, sizeof(reply.status.addr) - 1, host);
break;
default:
reply = (struct ctl_msg){
@@ -423,5 +439,7 @@ gt_bind(int argc, char **argv)
perror("tun_set_persist");
}
ctl_delete(ctl_fd);
return 0;
}

View File

@@ -2,21 +2,15 @@
#include "ctl.h"
#include "str.h"
#include <stdio.h>
#include <unistd.h>
#include <sys/socket.h>
#include <sys/stat.h>
#include <sys/un.h>
int
ctl_init(const char *dir, const char *file)
static int
ctl_setsun(struct sockaddr_un *dst, const char *dir, const char *file)
{
if (str_empty(dir) || str_empty(file)) {
errno = EINVAL;
return -1;
}
if (mkdir(dir, 0700) == -1 && errno != EEXIST)
return -1;
struct sockaddr_un sun = {
.sun_family = AF_UNIX,
};
@@ -31,14 +25,74 @@ ctl_init(const char *dir, const char *file)
}
}
*dst = sun;
return 0;
}
static int
ctl_bind(int fd, const char *dir, const char *file)
{
char tmp[32];
struct sockaddr_un sun;
if (str_empty(file)) {
for (int i = 0; i < 64; i++) {
if (snprintf(tmp, sizeof(tmp), ".%i", i) >= sizeof(tmp))
return -1;
if (ctl_setsun(&sun, dir, tmp))
return -1;
if (!bind(fd, (struct sockaddr *)&sun, sizeof(sun)))
return 0;
}
} else {
if (ctl_setsun(&sun, dir, file))
return -1;
unlink(sun.sun_path);
if (!bind(fd, (struct sockaddr *)&sun, sizeof(sun)))
return 0;
}
return -1;
}
void
ctl_delete(int fd)
{
if (fd == -1)
return;
struct sockaddr_storage ss;
socklen_t sslen = sizeof(ss);
if ((getsockname(fd, (struct sockaddr *)&ss, &sslen) == 0) &&
(ss.ss_family == AF_UNIX))
unlink(((struct sockaddr_un *)&ss)->sun_path);
close(fd);
}
int
ctl_create(const char *dir, const char *file)
{
if (str_empty(dir)) {
errno = EINVAL;
return -1;
}
if (mkdir(dir, 0700) == -1 && errno != EEXIST)
return -1;
int fd = socket(AF_UNIX, SOCK_DGRAM, 0);
if (fd == -1)
return -1;
unlink(sun.sun_path);
if (bind(fd, (struct sockaddr *)&sun, sizeof(sun)) == -1) {
if (ctl_bind(fd, dir, file)) {
int err = errno;
close(fd);
errno = err;
@@ -56,19 +110,10 @@ ctl_connect(int fd, const char *dir, const char *file)
return -1;
}
struct sockaddr_un sun = {
.sun_family = AF_UNIX,
};
struct sockaddr_un sun;
const char *path[] = {dir, "/", file};
const size_t len = sizeof(sun.sun_path) - 1;
if (str_cat(sun.sun_path, len, path, COUNT(path)) == len) {
if (str_cat(NULL, len + 1, path, COUNT(path)) > len) {
errno = EINVAL;
return -1;
}
}
if (ctl_setsun(&sun, dir, file))
return -1;
return connect(fd, (struct sockaddr *)&sun, sizeof(sun));
}

View File

@@ -4,7 +4,8 @@ enum ctl_type {
CTL_UNKNOWN,
CTL_PATH_ADD,
CTL_PATH_DEL,
CTL_PING,
CTL_STATUS,
CTL_STATUS_REPLY,
CTL_REPLY,
};
@@ -19,9 +20,20 @@ struct ctl_msg {
char addr[256];
} add, del;
} path;
struct {
size_t mtu;
int mtu_auto;
int chacha;
char addr[256];
unsigned short port;
unsigned short bind_port;
int ipv4;
int ipv6;
} status;
int reply;
};
};
int ctl_init (const char *, const char *);
int ctl_connect (int, const char *, const char *);
int ctl_create (const char *, const char *);
int ctl_connect (int, const char *, const char *);
void ctl_delete (int);

View File

@@ -50,20 +50,7 @@ gt_version(int argc, char **argv)
return 0;
}
int
gt_show(int argc, char **argv)
{
printf("show (todo)\n");
return 0;
}
int
gt_key(int argc, char **argv)
{
printf("key (todo)\n");
return 0;
}
int gt_show(int, char **);
int gt_bind(int, char **);
int gt_path(int, char **);
int gt_keygen(int, char **);

View File

@@ -28,13 +28,11 @@ gt_path(int argc, char **argv)
struct ctl_msg msg;
if (argz_is_set(pathz, "up")) {
gt_log("up\n");
msg = (struct ctl_msg){
.type = CTL_PATH_ADD,
};
str_cpy(msg.path.add.addr, sizeof(msg.path.add.addr) - 1, addr);
} else if (argz_is_set(pathz, "down")) {
gt_log("down\n");
msg = (struct ctl_msg){
.type = CTL_PATH_DEL,
};
@@ -44,27 +42,25 @@ gt_path(int argc, char **argv)
return 0;
}
int ctl_fd = ctl_init("/run/" PACKAGE_NAME, "client");
int fd = ctl_create("/run/" PACKAGE_NAME, NULL);
if (ctl_fd == -1) {
perror("ctl_init");
if (fd == -1) {
perror("ctl_create");
return 1;
}
if (ctl_connect(ctl_fd, "/run/" PACKAGE_NAME, dev) == -1) {
if (ctl_connect(fd, "/run/" PACKAGE_NAME, dev) == -1) {
gt_log("couldn't connect to %s\n", dev);
return 1;
}
if (send(ctl_fd, &msg, sizeof(msg), 0) == -1) {
perror("send");
ctl_delete(fd);
return 1;
}
struct ctl_msg reply;
if (recv(ctl_fd, &reply, sizeof(reply), 0) == -1) {
perror("recv");
if ((send(fd, &msg, sizeof(msg), 0) == -1) ||
(recv(fd, &reply, sizeof(reply), 0) == -1)) {
perror("send/recv");
ctl_delete(fd);
return 1;
}
@@ -82,7 +78,7 @@ gt_path(int argc, char **argv)
gt_log("bad reply from server: %i\n", reply.type);
}
close(ctl_fd);
ctl_delete(fd);
return 0;
}

99
src/show.c Normal file
View File

@@ -0,0 +1,99 @@
#include "common.h"
#include "ctl.h"
#include "str.h"
#include <stdio.h>
#include <sys/socket.h>
#include <dirent.h>
#include <sys/un.h>
static int
gt_show_tunnel(int fd, const char *dev)
{
if (ctl_connect(fd, "/run/" PACKAGE_NAME, dev) == -1) {
perror("connect");
return -1;
}
struct ctl_msg reply, msg = {
.type = CTL_STATUS,
};
if ((send(fd, &msg, sizeof(msg), 0) == -1) ||
(recv(fd, &reply, sizeof(reply), 0) == -1)) {
perror("send/recv");
return -1;
}
if (reply.type != CTL_STATUS_REPLY)
return -1;
if (str_empty(reply.status.addr)) {
printf("server %s:\n"
" mtu: %zu\n"
" auto mtu: %s\n"
" bind port: %hu\n"
" cipher: %s\n"
" ipv4: %s\n"
" ipv6: %s\n",
dev,
reply.status.mtu,
reply.status.mtu_auto ? "enabled" : "disabled",
reply.status.bind_port,
reply.status.chacha ? "chacha20poly1305" : "aes256gcm",
reply.status.ipv4 ? "enabled" : "disabled",
reply.status.ipv6 ? "enabled" : "disabled");
} else {
printf("client %s:\n"
" host: %s\n"
" port: %hu\n"
" mtu: %zu\n"
" auto mtu: %s\n"
" bind port: %hu\n"
" cipher: %s\n"
" ipv4: %s\n"
" ipv6: %s\n",
dev,
reply.status.addr, reply.status.port,
reply.status.mtu,
reply.status.mtu_auto ? "enabled" : "disabled",
reply.status.bind_port,
reply.status.chacha ? "chacha20poly1305" : "aes256gcm",
reply.status.ipv4 ? "enabled" : "disabled",
reply.status.ipv6 ? "enabled" : "disabled");
}
return 0;
}
int
gt_show(int argc, char **argv)
{
DIR *dp = opendir("/run/" PACKAGE_NAME);
if (!dp) {
perror("opendir");
return 1;
}
struct dirent *d = NULL;
while (d = readdir(dp), d) {
if (d->d_name[0] == '.')
continue;
int fd = ctl_create("/run/" PACKAGE_NAME, NULL);
if (fd == -1) {
perror("ctl_create");
return 1;
}
gt_show_tunnel(fd, d->d_name);
ctl_delete(fd);
}
closedir(dp);
return 0;
}