mirror of
https://gitlab.nic.cz/labs/bird.git
synced 2025-01-11 03:21:53 +00:00
632 lines
12 KiB
C
632 lines
12 KiB
C
/*
|
|
* BIRD Client
|
|
*
|
|
* (c) 1999--2004 Martin Mares <mj@ucw.cz>
|
|
* (c) 2013 Tomas Hlavacek <tmshlvck@gmail.com>
|
|
*
|
|
* Can be freely distributed and used under the terms of the GNU GPL.
|
|
*/
|
|
|
|
/**
|
|
* DOC: BIRD client
|
|
*
|
|
* There are two variants of BIRD client: regular and light. regular
|
|
* variant depends on readline and ncurses libraries, while light
|
|
* variant uses just libc. Most of the code and the main() is common
|
|
* for both variants (in client.c file) and just a few functions are
|
|
* different (in birdc.c for regular and birdcl.c for light). Two
|
|
* binaries are generated by linking common object files like client.o
|
|
* (which is compiled from client.c just once) with either birdc.o or
|
|
* birdcl.o for each variant.
|
|
*/
|
|
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#include <fcntl.h>
|
|
#include <unistd.h>
|
|
#include <errno.h>
|
|
#include <sys/select.h>
|
|
#include <sys/socket.h>
|
|
#include <sys/types.h>
|
|
#include <sys/un.h>
|
|
|
|
#include "nest/bird.h"
|
|
#include "lib/resource.h"
|
|
#include "lib/string.h"
|
|
#include "client/client.h"
|
|
#include "client/print_cbor.c"
|
|
#include "sysdep/unix/unix.h"
|
|
|
|
#define SERVER_READ_BUF_LEN 4096
|
|
|
|
|
|
static char *opt_list = "s:vrl";
|
|
static int verbose, restricted, once;
|
|
static char *init_cmd;
|
|
|
|
static char *server_path = PATH_CONTROL_SOCKET;
|
|
static int server_fd;
|
|
static byte server_read_buf[SERVER_READ_BUF_LEN];
|
|
static byte *server_read_pos = server_read_buf;
|
|
|
|
int init = 1; /* During intial sequence */
|
|
int busy = 1; /* Executing BIRD command */
|
|
int interactive; /* Whether stdin is terminal */
|
|
int last_code; /* Last return code */
|
|
int cbor_mode; /* Convert to cbor and push to yi socket (and convert answer back) */
|
|
|
|
static int num_lines, skip_input;
|
|
int term_lns, term_cls;
|
|
|
|
|
|
/*** Parsing of arguments ***/
|
|
|
|
static void
|
|
usage(char *name)
|
|
{
|
|
fprintf(stderr, "Usage: %s [-s <control-socket>] [-v] [-r] [-l]\n", name);
|
|
exit(1);
|
|
}
|
|
|
|
static void
|
|
parse_args(int argc, char **argv)
|
|
{
|
|
int server_changed = 0;
|
|
int c;
|
|
|
|
while ((c = getopt(argc, argv, opt_list)) >= 0)
|
|
switch (c)
|
|
{
|
|
case 's':
|
|
server_path = optarg;
|
|
server_changed = 1;
|
|
break;
|
|
case 'v':
|
|
verbose++;
|
|
break;
|
|
case 'r':
|
|
restricted = 1;
|
|
break;
|
|
case 'l':
|
|
if (!server_changed)
|
|
server_path = xbasename(server_path);
|
|
break;
|
|
default:
|
|
usage(argv[0]);
|
|
}
|
|
|
|
/* If some arguments are not options, we take it as commands */
|
|
if (optind < argc)
|
|
{
|
|
char *tmp;
|
|
int i;
|
|
int len = 0;
|
|
|
|
for (i = optind; i < argc; i++)
|
|
len += strlen(argv[i]) + 1;
|
|
|
|
tmp = init_cmd = malloc(len);
|
|
for (i = optind; i < argc; i++)
|
|
{
|
|
strcpy(tmp, argv[i]);
|
|
tmp += strlen(tmp);
|
|
*tmp++ = ' ';
|
|
}
|
|
tmp[-1] = 0;
|
|
|
|
once = 1;
|
|
interactive = 0;
|
|
}
|
|
}
|
|
|
|
|
|
/*** Input ***/
|
|
|
|
static void server_send(char *cmd);
|
|
static void
|
|
server_send_byte(byte *cmd, int l);
|
|
|
|
static int
|
|
handle_internal_command(char *cmd)
|
|
{
|
|
if (!strncmp(cmd, "exit", 4) || !strncmp(cmd, "quit", 4))
|
|
{
|
|
cleanup();
|
|
exit(0);
|
|
}
|
|
if (!strncmp(cmd, "help", 4))
|
|
{
|
|
puts("Press `?' for context sensitive help.");
|
|
return 1;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
uint compare_string(byte *str1, uint length, const char *str2) {
|
|
if (length < strlen(str2)) {
|
|
return 0;
|
|
}
|
|
for (size_t i = 0; i < strlen(str2); i++) {
|
|
if (str1[i]!=str2[i]) {
|
|
return 0;
|
|
}
|
|
}
|
|
return 1;
|
|
}
|
|
|
|
void write_args_cbor(char *cmd_buffer, struct cbor_writer *w)
|
|
{
|
|
size_t l = strlen(cmd_buffer);
|
|
size_t pt = 0;
|
|
while (cmd_buffer[pt] == ' ')
|
|
pt++;
|
|
if (pt+1 >= l)
|
|
{
|
|
return;
|
|
}
|
|
cbor_add_string(w, "args");
|
|
cbor_open_list(w);
|
|
while (l>pt)
|
|
{
|
|
size_t next_space = 0;
|
|
while (next_space + pt < l && cmd_buffer[next_space + pt] != ' ' && l>pt)
|
|
next_space++;
|
|
cbor_open_block_with_length(w, 1);
|
|
cbor_add_string(w, "arg");
|
|
cbor_nonterminated_string(w, &cmd_buffer[pt], next_space);
|
|
pt += next_space;
|
|
while (cmd_buffer[pt] == ' ')
|
|
pt++;
|
|
}
|
|
cbor_close_block_or_list(w);
|
|
}
|
|
|
|
void
|
|
make_cmd_cbor(char *cmd_buffer)
|
|
{
|
|
size_t l = strlen(cmd_buffer);
|
|
char cbor_buf[l*10];
|
|
struct linpool *lp = lp_new(&root_pool);
|
|
|
|
struct cbor_writer *w = cbor_init(cbor_buf, l*10, lp);
|
|
cbor_open_block_with_length(w, 1);
|
|
cbor_add_string(w, "command:do");
|
|
|
|
char *show = "show ";
|
|
int buf_pt = 0;
|
|
if (compare_string(cmd_buffer, l, show))
|
|
{
|
|
cbor_open_block(w);
|
|
buf_pt += strlen(show);
|
|
l -= strlen(show);
|
|
if (compare_string(&cmd_buffer[buf_pt], l, "memory"))
|
|
{
|
|
cbor_string_int(w, "command", SHOW_MEMORY);
|
|
cbor_close_block_or_list(w);
|
|
server_send_byte(cbor_buf, w->pt);
|
|
lp_flush(lp);
|
|
return;
|
|
}
|
|
else if (compare_string(&cmd_buffer[buf_pt], l, "status"))
|
|
{
|
|
cbor_string_int(w, "command", SHOW_STATUS);
|
|
cbor_close_block_or_list(w);
|
|
cbor_write_to_file(w, "status_command.cbor");
|
|
server_send_byte(cbor_buf, w->pt);
|
|
lp_flush(lp);
|
|
return;
|
|
}
|
|
else if (compare_string(&cmd_buffer[buf_pt], l, "symbols"))
|
|
{
|
|
cbor_string_int(w, "command", SHOW_SYMBOLS);
|
|
write_args_cbor(&cmd_buffer[buf_pt + strlen("symbols ")], w);
|
|
cbor_close_block_or_list(w);
|
|
server_send_byte(cbor_buf, w->pt);
|
|
lp_flush(lp);
|
|
return;
|
|
}
|
|
else if (compare_string(&cmd_buffer[buf_pt], l, "ospf"))
|
|
{
|
|
cbor_string_int(w, "command", SHOW_OSPF);
|
|
write_args_cbor(&cmd_buffer[buf_pt + strlen("ospf")], w);
|
|
cbor_close_block_or_list(w);
|
|
server_send_byte(cbor_buf, w->pt);
|
|
lp_flush(lp);
|
|
return;
|
|
}
|
|
else
|
|
{
|
|
printf("this command is not implemented yet\n");
|
|
}
|
|
|
|
}
|
|
else if (compare_string(cmd_buffer, l, "down"))
|
|
{
|
|
cbor_add_string(w, "down");
|
|
server_send_byte(cbor_buf, w->pt);
|
|
die("Shutdown from client");
|
|
return;
|
|
}
|
|
lp_flush(lp);
|
|
printf("this command is not implemented yet\n");
|
|
}
|
|
|
|
static void
|
|
submit_server_command(char *cmd)
|
|
{
|
|
if (cbor_mode)
|
|
{
|
|
make_cmd_cbor(cmd);
|
|
return;
|
|
}
|
|
|
|
busy = 1;
|
|
num_lines = 2;
|
|
fprintf(stderr, "cmd: %s \n", cmd);
|
|
server_send(cmd);
|
|
}
|
|
|
|
static inline void
|
|
submit_init_command(char *cmd_raw)
|
|
{
|
|
char *cmd = cmd_expand(cmd_raw);
|
|
|
|
if (!cmd)
|
|
{
|
|
cleanup();
|
|
exit(1);
|
|
}
|
|
|
|
submit_server_command(cmd);
|
|
free(cmd);
|
|
}
|
|
|
|
void
|
|
submit_command(char *cmd_raw)
|
|
{
|
|
char *cmd = cmd_expand(cmd_raw);
|
|
|
|
if (!cmd)
|
|
return;
|
|
|
|
if (!handle_internal_command(cmd))
|
|
submit_server_command(cmd);
|
|
|
|
free(cmd);
|
|
}
|
|
|
|
|
|
static void
|
|
init_commands(void)
|
|
{
|
|
if (restricted)
|
|
{
|
|
submit_server_command("restrict");
|
|
restricted = 0;
|
|
return;
|
|
}
|
|
|
|
if (init_cmd)
|
|
{
|
|
/* First transition - client received hello from BIRD
|
|
and there is waiting initial command */
|
|
submit_init_command(init_cmd);
|
|
init_cmd = NULL;
|
|
return;
|
|
}
|
|
|
|
if (once)
|
|
{
|
|
/* Initial command is finished and we want to exit */
|
|
cleanup();
|
|
exit((last_code < 8000) ? 0 : 1);
|
|
}
|
|
|
|
input_init();
|
|
|
|
term_lns = (term_lns > 0) ? term_lns : 25;
|
|
term_cls = (term_cls > 0) ? term_cls : 80;
|
|
|
|
init = 0;
|
|
}
|
|
|
|
|
|
/*** Output ***/
|
|
|
|
void
|
|
more(void)
|
|
{
|
|
more_begin();
|
|
printf("--More--\015");
|
|
fflush(stdout);
|
|
|
|
redo:
|
|
switch (getchar())
|
|
{
|
|
case ' ':
|
|
num_lines = 2;
|
|
break;
|
|
case '\n':
|
|
case '\r':
|
|
num_lines--;
|
|
break;
|
|
case 'q':
|
|
skip_input = 1;
|
|
break;
|
|
default:
|
|
goto redo;
|
|
}
|
|
|
|
printf(" \015");
|
|
fflush(stdout);
|
|
more_end();
|
|
}
|
|
|
|
|
|
/*** Communication with server ***/
|
|
|
|
static void
|
|
server_connect(void)
|
|
{
|
|
struct sockaddr_un sa;
|
|
|
|
server_fd = socket(AF_UNIX, SOCK_STREAM, 0);
|
|
if (server_fd < 0)
|
|
DIE("Cannot create socket");
|
|
|
|
if (strlen(server_path) >= sizeof(sa.sun_path))
|
|
die("server_connect: path too long");
|
|
|
|
bzero(&sa, sizeof(sa));
|
|
sa.sun_family = AF_UNIX;
|
|
strcpy(sa.sun_path, server_path);
|
|
if (connect(server_fd, (struct sockaddr *) &sa, SUN_LEN(&sa)) < 0)
|
|
DIE("Unable to connect to server control socket (%s)", server_path);
|
|
if (fcntl(server_fd, F_SETFL, O_NONBLOCK) < 0)
|
|
DIE("fcntl");
|
|
fprintf(stdout, "Socket: %s connected ok\n", server_path);
|
|
}
|
|
|
|
|
|
#define PRINTF(LEN, PARGS...) do { if (!skip_input) len = printf(PARGS); } while(0)
|
|
|
|
static void
|
|
server_got_reply(char *x)
|
|
{
|
|
int code;
|
|
int len = 0;
|
|
|
|
if (*x == '+') /* Async reply */
|
|
PRINTF(len, ">>> %s\n", x+1);
|
|
else if (x[0] == ' ') /* Continuation */
|
|
PRINTF(len, "%s%s\n", verbose ? " " : "", x+1);
|
|
else if (strlen(x) > 4 &&
|
|
sscanf(x, "%d", &code) == 1 && code >= 0 && code < 10000 &&
|
|
(x[4] == ' ' || x[4] == '-'))
|
|
{
|
|
if (code)
|
|
PRINTF(len, "%s\n", verbose ? x : x+5);
|
|
|
|
last_code = code;
|
|
|
|
if (x[4] == ' ')
|
|
{
|
|
busy = 0;
|
|
skip_input = 0;
|
|
return;
|
|
}
|
|
}
|
|
else
|
|
PRINTF(len, "??? <%s>\n", x);
|
|
|
|
if (interactive && busy && !skip_input && !init && (len > 0))
|
|
{
|
|
num_lines += (len + term_cls - 1) / term_cls; /* Divide and round up */
|
|
if (num_lines >= term_lns)
|
|
more();
|
|
}
|
|
}
|
|
|
|
void
|
|
server_got_binary(int c)
|
|
{
|
|
// TODO check cbor hello
|
|
if (cbor_mode == 0)
|
|
{
|
|
byte expected[] = {0x87, 0x42, 0x49, 0x52, 0x44, 0x0D, 0x0A, 0x1A, 0x0A, 0x01};
|
|
if (c < 10)
|
|
die("too short header");
|
|
for (int i = 0; i < 9; i++)
|
|
{
|
|
if (server_read_buf[i] != expected[i])
|
|
die("wrong header");
|
|
}
|
|
if (server_read_buf[9] != expected[9])
|
|
die("unknown version of binary communication");
|
|
cbor_mode = 1;
|
|
}
|
|
else {
|
|
print_cbor_response(server_read_buf, c);
|
|
}
|
|
busy = 0;
|
|
skip_input = 0;
|
|
}
|
|
|
|
static void
|
|
server_read(void)
|
|
{
|
|
int c;
|
|
byte *start, *p;
|
|
|
|
redo:
|
|
c = read(server_fd, server_read_pos, server_read_buf + sizeof(server_read_buf) - server_read_pos);
|
|
if (!c)
|
|
die("Connection closed by server");
|
|
if (c < 0)
|
|
{
|
|
if (errno == EINTR)
|
|
goto redo;
|
|
else
|
|
DIE("Server read error");
|
|
}
|
|
|
|
if ((init && (*server_read_buf == 0x87)) || cbor_mode)
|
|
return server_got_binary(c);
|
|
|
|
start = server_read_buf;
|
|
p = server_read_pos;
|
|
server_read_pos += c;
|
|
while (p < server_read_pos)
|
|
if (*p++ == '\n')
|
|
{
|
|
p[-1] = 0;
|
|
server_got_reply(start);
|
|
start = p;
|
|
}
|
|
if (start != server_read_buf)
|
|
{
|
|
int l = server_read_pos - start;
|
|
memmove(server_read_buf, start, l);
|
|
server_read_pos = server_read_buf + l;
|
|
}
|
|
else if (server_read_pos == server_read_buf + sizeof(server_read_buf))
|
|
{
|
|
strcpy(server_read_buf, "?<too-long>");
|
|
server_read_pos = server_read_buf + 11;
|
|
}
|
|
}
|
|
|
|
static void
|
|
select_loop(void)
|
|
{
|
|
int rv;
|
|
while (1)
|
|
{
|
|
if (init && !busy)
|
|
init_commands();
|
|
|
|
if (!init)
|
|
input_notify(!busy);
|
|
|
|
fd_set select_fds;
|
|
FD_ZERO(&select_fds);
|
|
|
|
FD_SET(server_fd, &select_fds);
|
|
if (!busy)
|
|
FD_SET(0, &select_fds);
|
|
|
|
rv = select(server_fd+1, &select_fds, NULL, NULL, NULL);
|
|
if (rv < 0)
|
|
{
|
|
if (errno == EINTR)
|
|
continue;
|
|
else
|
|
DIE("select");
|
|
}
|
|
|
|
if (FD_ISSET(0, &select_fds))
|
|
{
|
|
input_read();
|
|
continue;
|
|
}
|
|
|
|
if (FD_ISSET(server_fd, &select_fds))
|
|
{
|
|
server_read();
|
|
continue;
|
|
}
|
|
}
|
|
}
|
|
|
|
static void
|
|
wait_for_write(int fd)
|
|
{
|
|
while (1)
|
|
{
|
|
int rv;
|
|
fd_set set;
|
|
FD_ZERO(&set);
|
|
FD_SET(fd, &set);
|
|
|
|
rv = select(fd+1, NULL, &set, NULL, NULL);
|
|
if (rv < 0)
|
|
{
|
|
if (errno == EINTR)
|
|
continue;
|
|
else
|
|
DIE("select");
|
|
}
|
|
|
|
if (FD_ISSET(server_fd, &set))
|
|
return;
|
|
}
|
|
}
|
|
|
|
static void
|
|
server_send(char *cmd)
|
|
{
|
|
int l = strlen(cmd);
|
|
byte *z = alloca(l + 1);
|
|
|
|
memcpy(z, cmd, l);
|
|
z[l++] = '\n';
|
|
while (l)
|
|
{
|
|
int cnt = write(server_fd, z, l);
|
|
|
|
if (cnt < 0)
|
|
{
|
|
if (errno == EAGAIN)
|
|
wait_for_write(server_fd);
|
|
else if (errno == EINTR)
|
|
continue;
|
|
else
|
|
DIE("Server write error");
|
|
}
|
|
else
|
|
{
|
|
l -= cnt;
|
|
z += cnt;
|
|
}
|
|
}
|
|
}
|
|
|
|
static void
|
|
server_send_byte(byte *cmd, int l)
|
|
{
|
|
byte *z = alloca(l);
|
|
|
|
memcpy(z, cmd, l);
|
|
while (l)
|
|
{
|
|
int cnt = write(server_fd, z, l);
|
|
|
|
if (cnt < 0)
|
|
{
|
|
if (errno == EAGAIN)
|
|
wait_for_write(server_fd);
|
|
else if (errno == EINTR)
|
|
continue;
|
|
else
|
|
DIE("Server write error");
|
|
}
|
|
else
|
|
{
|
|
l -= cnt;
|
|
z += cnt;
|
|
}
|
|
}
|
|
}
|
|
|
|
int
|
|
main(int argc, char **argv)
|
|
{
|
|
interactive = isatty(0);
|
|
parse_args(argc, argv);
|
|
cmd_build_tree();
|
|
resource_init();
|
|
server_connect();
|
|
select_loop();
|
|
return 0;
|
|
}
|