0
0
mirror of https://gitlab.nic.cz/labs/bird.git synced 2025-01-18 15:01:53 +00:00
bird/nest/cbor_cmds.c
2024-01-12 10:09:45 +01:00

369 lines
11 KiB
C

#include "nest/bird.h"
#include "nest/protocol.h"
#include "nest/route.h"
#include "nest/cli.h"
#include "conf/conf.h"
#include "lib/string.h"
#include "filter/filter.h"
#include "nest/cbor_cmds.h"
#include "proto/ospf/ospf_for_cbor.c"
int64_t preprocess_time(btime t) {
return tm_get_real_time(t) TO_S ;
}
uint compare_byte_str(byte *str1, uint length, const char *str2) {
if (length != strlen(str2)) {
return 0;
}
for (size_t i = 0; i < length; i++) {
if (str1[i]!=str2[i]) {
return 0;
}
}
return 1;
}
static char *
proto_state_name_stolen_for_cbor(struct proto *p)
{
switch (p->proto_state)
{
case PS_DOWN: return p->active ? "flush" : "down";
case PS_START: return "start";
case PS_UP: return "up";
case PS_STOP: return "stop";
default: return "???";
}
}
uint
cmd_show_protocols_cbor(byte *tbuf, uint capacity, struct arg_list *args, struct linpool *lp)
{
log("in cmd_show_protocols_cbor");
struct cbor_writer *w = cbor_init(tbuf, capacity, lp);
cbor_open_block_with_length(w, 1);
cbor_add_string(w, "show_protocols:message");
cbor_open_block_with_length(w, 1);
cbor_add_string(w, "table");
cbor_open_list(w);
int all = 0;
int protocol = -1;
if (args->pt > 0 && compare_byte_str(args->args[0].arg, args->args[0].len, "all"))
{
all = 1;
}
if (args->pt - all > 0)
{
protocol = all;
}
struct proto *p;
WALK_LIST(p, proto_list)
{
if (protocol == -1 || compare_byte_str(args->args[protocol].arg, args->args[protocol].len, p->name))
{
cbor_open_block(w);
cbor_string_string(w, "name", p->name);
cbor_string_string(w, "proto", p->proto->name);
cbor_string_string(w, "table", p->main_channel ? p->main_channel->table->name : "---");
cbor_string_string(w, "state", proto_state_name_stolen_for_cbor(p));
cbor_string_epoch_time(w, "since", tm_get_real_time(p->last_state_change), -6);
byte buf[256];
buf[0] = 0;
if (p->proto->get_status)
p->proto->get_status(p, buf);
cbor_string_string(w, "info", buf);
if (all)
{
if (p->cf->dsc)
cbor_string_string(w, "description", p->cf->dsc);
if (p->message)
cbor_string_string(w, "message", p->message);
if (p->cf->router_id)
cbor_string_ipv4(w, "router_id", p->cf->router_id);
if (p->vrf_set)
cbor_string_string(w, "vrf", p->vrf ? p->vrf->name : "default");
if (p->proto->show_proto_info_cbor)
p->proto->show_proto_info_cbor(w, p);
else
{
struct channel *c;
WALK_LIST(c, p->channels)
channel_show_info_cbor(w, c);
}
}
cbor_close_block_or_list(w);
}
}
cbor_close_block_or_list(w);
return w->pt;
}
extern pool *rt_table_pool;
extern pool *rta_pool;
extern uint *pages_kept;
uint
cmd_show_memory_cbor(byte *tbuf, uint capacity, struct linpool *lp)
{
log("in cmd_show_memory_cbor");
struct cbor_writer *w = cbor_init(tbuf, capacity, lp);
log("w->pt %i w->cbor %i", w->pt, w->cbor);
cbor_open_block_with_length(w, 1);
cbor_add_string(w, "show_memory:message");
cbor_open_block_with_length(w, 2);
cbor_string_string(w, "header", "BIRD memory usage");
cbor_add_string(w, "body");
cbor_open_block(w);
struct resmem memory = rmemsize(rt_table_pool);
cbor_named_block_two_ints(w, "routing_tables", "effective", memory.effective, "overhead", memory.overhead);
memory = rmemsize(rta_pool);
cbor_named_block_two_ints(w, "route_attributes", "effective", memory.effective, "overhead", memory.overhead);
memory = rmemsize(proto_pool);
cbor_named_block_two_ints(w, "protocols", "effective", memory.effective, "overhead", memory.overhead);
memory = rmemsize(config_pool);
cbor_named_block_two_ints(w, "current_config", "effective", memory.effective, "overhead", memory.overhead);
memory = rmemsize(&root_pool);
#ifdef HAVE_MMAP
cbor_named_block_two_ints(w, "standby_memory", "effective", 0, "overhead", page_size * *pages_kept);
#endif
memory.overhead += page_size * *pages_kept;
cbor_named_block_two_ints(w, "total", "effective", memory.effective, "overhead", memory.overhead);
cbor_close_block_or_list(w); // we do not know for sure, that standby memory will be printed, so we do not know number of block items. If we know that, we open the block for 6 (or 5) items and we do not close anything
cbor_write_to_file(w, "show_memory.cbor");
log("show memory written");
return w->pt;
}
extern int shutting_down;
extern int configuring;
uint
cmd_show_status_cbor(byte *tbuf, uint capacity, struct linpool *lp)
{
log("show status");
struct cbor_writer *w = cbor_init(tbuf, capacity, lp);
cbor_open_block_with_length(w, 1);
cbor_add_string(w, "show_status:message");
cbor_open_block_with_length(w, 3);
cbor_string_string(w, "version", BIRD_VERSION);
cbor_add_string(w, "body");
cbor_open_block(w);
cbor_string_ipv4(w, "router_id", config->router_id);
cbor_string_string(w, "hostname", config->hostname);
cbor_string_epoch_time(w, "server_time", tm_get_real_time(current_time()), -6);
cbor_string_epoch_time(w, "last_reboot", tm_get_real_time(boot_time), -6);
cbor_string_epoch_time(w, "last_reconfiguration", tm_get_real_time(config->load_time), -6);
if (is_gr_active())
{
cbor_add_string(w, "gr_restart");
cbor_open_block_with_length(w, 2);
cbor_string_int(w, "waiting_for_n_channels_to_recover", get_graceful_restart_locks_num());
cbor_add_string(w, "wait_timer");
cbor_open_block_with_length(w, 2);
cbor_string_relativ_time(w, "remains", get_tm_remains_gr_wait_timer(), -6);
cbor_string_relativ_time(w, "count_time", get_config_gr_wait(), -6);
}
cbor_close_block_or_list(w);
cbor_add_string(w, "state");
if (shutting_down)
cbor_add_string(w, "Shutdown in progress");
else if (configuring)
cbor_add_string(w, "Reconfiguration in progress");
else
cbor_add_string(w, "Daemon is up and running");
cbor_write_to_file(w, "test.cbor");
return w->pt;
}
int parse_show_symbols_arg(struct argument *argument)
{
char *params[] = {"table", "filter", "function", "protocol", "template", "constant", "variable"};
int param_vals[] = {SYM_TABLE, SYM_FILTER, SYM_FUNCTION, SYM_PROTO, SYM_TEMPLATE, SYM_CONSTANT, SYM_VARIABLE}; // defined in conf.h
for (size_t j = 0; j < sizeof(params)/sizeof(char*); j++)
{
if (compare_byte_str(argument->arg, argument->len, params[j]))
{
return param_vals[j];
}
}
return -1;
}
uint
cmd_show_symbols_cbor(byte *tbuf, uint capacity, struct arg_list *args, struct linpool *lp)
{
struct cbor_writer *w = cbor_init(tbuf, capacity, lp);
cbor_open_block_with_length(w, 1);
cbor_add_string(w, "show_symbols:message");
cbor_open_block_with_length(w, 1);
int show_type = SYM_VOID;
if (args->pt > 0)
{
show_type = parse_show_symbols_arg(&args->args[args->pt - 1]); // Takes just the last one argument. Current bird cli answers only last argument too, but can fail on previous.
}
if (show_type == -1)
{
cbor_add_string(w, "table");
cbor_open_list_with_length(w, 1);
cbor_open_block_with_length(w, 2);
for (const struct sym_scope *scope = config->root_scope; scope; scope = scope->next)
{
HASH_WALK(scope->hash, next, sym)
{
if (compare_byte_str(args->args[args->pt - 1].arg, args->args[args->pt - 1].len, sym->name))
{
cbor_add_string(w, "name");
cbor_nonterminated_string(w, args->args[args->pt - 1].arg, args->args[args->pt - 1].len);
cbor_string_string(w, "type", cf_symbol_class_name(sym));
return w->pt;
}
}
HASH_WALK_END;
}
cbor_add_string(w, "name");
cbor_nonterminated_string(w, args->args[args->pt - 1].arg, args->args[args->pt - 1].len);
cbor_string_string(w, "type", "symbol not known");
return w->pt;
}
else
{
cbor_add_string(w, "table");
cbor_open_list(w);
for (const struct sym_scope *scope = config->root_scope; scope; scope = scope->next)
HASH_WALK(scope->hash, next, sym)
{
if (!sym->scope->active)
continue;
if (show_type == SYM_VARIABLE || show_type == SYM_CONSTANT)
{
if (!(show_type == (int)(sym->class & 0xffffff00)))
continue;
}
else if (show_type != SYM_VOID && (sym->class != show_type))
continue;
cbor_open_block_with_length(w, 2);
cbor_string_string(w, "name", sym->name);
cbor_string_string(w, "type", cf_symbol_class_name(sym));
}
HASH_WALK_END;
cbor_close_block_or_list(w);
return w->pt;
}
}
struct proto *
cbor_get_proto_type(enum protocol_class proto_type, struct cbor_writer *w)
{
log("in type");
struct proto *p, *q;
p = NULL;
WALK_LIST(q, proto_list)
if ((q->proto->class == proto_type) && (q->proto_state != PS_DOWN))
{
if (p)
{
cbor_string_string(w, "error", "multiple protocols running");
return NULL;
}
p = q;
}
if (!p)
{
cbor_string_string(w, "error", "no such protocols running");
return NULL;
}
return p;
}
struct proto *
cbor_get_proto_name(struct argument *arg, enum protocol_class proto_type, struct cbor_writer *w)
{
log("in name");
struct proto *q;
WALK_LIST(q, proto_list)
{
log("%s %s %i %i %i", arg->arg, q->name, compare_byte_str(arg->arg, arg->len, q->name) , (q->proto_state != PS_DOWN) , (q->proto->class == proto_type));
if (compare_byte_str(arg->arg, arg->len, q->name) && (q->proto_state != PS_DOWN) && (q->proto->class == proto_type))
{
return q;
}
}
cbor_add_string(w, "not found");
cbor_nonterminated_string(w, arg->arg, arg->len);
return NULL;
}
uint
cmd_show_ospf_cbor(byte *tbuf, uint capacity, struct arg_list *args, struct linpool *lp)
{
log("in ospf args %i, pt %i", args, args->pt);
struct cbor_writer *w = cbor_init(tbuf, capacity, lp);
cbor_open_block_with_length(w, 1);
cbor_add_string(w, "show_ospf:message");
if (args->pt == 0)
{
cbor_open_block_with_length(w, 1);
cbor_string_string(w, "not_implemented", "show everything about ospf");
return w->pt;
}
if (compare_byte_str(args->args[0].arg, args->args[0].len, "topology"))
{
cbor_open_block(w);
struct proto *proto;
int all_ospf = (args->pt > 1) && compare_byte_str(args->args[1].arg, args->args[1].len, "all");
if (args->pt - all_ospf > 1) // if there is protocol name
{
proto = cbor_get_proto_name(&args->args[args->pt -1], PROTOCOL_OSPF, w);
}
else {
proto = cbor_get_proto_type(PROTOCOL_OSPF, w);
}
if (proto == NULL)
{
cbor_close_block_or_list(w);
return w->pt;
}
ospf_sh_state_cbor(w, proto, 0, all_ospf);
cbor_close_block_or_list(w);
return w->pt;
} else {
cbor_open_block_with_length(w, 1);
cbor_add_string(w, "not_implemented");
cbor_nonterminated_string(w, args->args[0].arg, args->args[0].len);
return w->pt;
}
}