You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
kamailio/src/modules/corex/corex_rpc.c

286 lines
7.1 KiB

/**
* Copyright (C) 2011 Daniel-Constantin Mierla (asipto.com)
*
* This file is part of Kamailio, a free SIP server.
*
* 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 <stdio.h>
#include <unistd.h>
#include <stdlib.h>
#include <string.h>
#include "../../core/dprint.h"
#include "../../core/ut.h"
#include "../../core/pt.h"
#include "../../core/cfg/cfg.h"
#include "../../core/cfg/cfg_ctx.h"
#include "../../core/socket_info.h"
#include "../../core/name_alias.h"
#include "../../core/mem/shm_mem.h"
#include "../../core/rpc.h"
#include "../../core/rpc_lookup.h"
static const char *corex_rpc_list_sockets_doc[2] = {
"List listening sockets", 0};
/*
* RPC command to list the listening sockets
*/
static void corex_rpc_list_sockets(rpc_t *rpc, void *ctx)
{
void *th;
void *ih;
struct socket_info *si;
struct socket_info **list;
struct addr_info *ai;
unsigned short proto;
proto = PROTO_UDP;
do {
list = get_sock_info_list(proto);
for(si = list ? *list : 0; si; si = si->next) {
/* add structure node */
if(rpc->add(ctx, "{", &th) < 0) {
rpc->fault(ctx, 500, "Internal error socket structure");
return;
}
if(rpc->struct_add(th, "sss{", "af", get_af_name(si->address.af),
"proto", get_valid_proto_name(proto), "name", si->name.s,
"addrlist", &ih)
< 0) {
rpc->fault(ctx, 500, "Internal error address list structure");
return;
}
if(rpc->struct_add(ih, "s", "addr", si->address_str.s) < 0) {
rpc->fault(ctx, 500, "Internal error address structure");
return;
}
if(si->addr_info_lst) {
for(ai = si->addr_info_lst; ai; ai = ai->next) {
if(rpc->struct_add(ih, "s", "addr", ai->address_str.s)
< 0) {
rpc->fault(ctx, 500,
"Internal error extra address structure");
return;
}
}
}
if(rpc->struct_add(th, "sssssss", "port", si->port_no_str.s,
"sockstr", si->sock_str.s ? si->sock_str.s : "-",
"mcast", si->flags & SI_IS_MCAST ? "yes" : "no",
"mhomed", si->flags & SI_IS_MHOMED ? "yes" : "no",
"virtual", si->flags & SI_IS_VIRTUAL ? "yes" : "no",
"sockname", si->sockname.s ? si->sockname.s : "-",
"advertise",
si->useinfo.sock_str.s ? si->useinfo.sock_str.s : "-")
< 0) {
rpc->fault(ctx, 500, "Internal error attrs structure");
return;
}
}
} while((proto = next_proto(proto)));
return;
}
static const char *corex_rpc_list_aliases_doc[2] = {"List socket aliases", 0};
/*
* RPC command to list the socket aliases
*/
static void corex_rpc_list_aliases(rpc_t *rpc, void *ctx)
{
void *th;
struct host_alias *a;
for(a = aliases; a; a = a->next) {
/* add structure node */
if(rpc->add(ctx, "{", &th) < 0) {
rpc->fault(ctx, 500, "Internal error alias structure");
return;
}
if(rpc->struct_add(th, "sSd", "proto", get_valid_proto_name(a->proto),
"addr", &a->alias, "port", a->port)
< 0) {
rpc->fault(ctx, 500, "Internal error alias attributes");
return;
}
}
return;
}
static const char *corex_rpc_shm_status_doc[2] = {
"Trigger shm status dump to syslog", 0};
/*
* RPC command to dump shm status to syslog
*/
static void corex_rpc_shm_status(rpc_t *rpc, void *ctx)
{
LM_DBG("printing shared memory status report\n");
shm_status();
}
static const char *corex_rpc_shm_summary_doc[2] = {
"Trigger shm summary dump to syslog", 0};
/*
* RPC command to dump shm summary to syslog
*/
static void corex_rpc_shm_summary(rpc_t *rpc, void *ctx)
{
LM_DBG("printing shared memory summary report\n");
shm_sums();
}
static const char *corex_rpc_pkg_summary_doc[2] = {
"Trigger shm summary dump to syslog", 0};
static cfg_ctx_t *_cfg_corex_ctx = NULL;
/*
* RPC command to dump pkg summary to syslog
*/
static void corex_rpc_pkg_summary(rpc_t *rpc, void *c)
{
str group = str_init("core");
str var = str_init("mem_dump_pkg");
str sel = {0, 0};
int i;
if(rpc->scan(c, "Sd", &sel, &i) < 2) {
rpc->fault(c, 400, "Selector and value not provided");
return;
}
if(sel.len != 3) {
rpc->fault(c, 500, "Unsupported selector");
return;
}
if(strncasecmp(sel.s, "idx", 3) == 0) {
if(i < 0 || i >= *process_count) {
rpc->fault(c, 500, "Index value out of range");
return;
}
i = pt[i].pid;
} else if(strncasecmp(sel.s, "pid", 3) != 0) {
rpc->fault(c, 500, "Unsupported selector type");
return;
}
if(cfg_set_now(
_cfg_corex_ctx, &group, NULL, &var, (void *)(long)i, CFG_VAR_INT)
!= 0) {
rpc->fault(c, 500, "Operation failed");
return;
}
}
static const char *corex_rpc_debug_doc[2] = {
"Control the global level of debug", 0};
/*
* RPC command to dump shm summary to syslog
*/
static void corex_rpc_debug(rpc_t *rpc, void *ctx)
{
int newdbg = 0;
int olddbg = 0;
int setdbg = 0;
str gname = {"core", 4};
str vname = {"debug", 5};
void *vval = 0;
unsigned int vtype;
void *th;
if(rpc->scan(ctx, "*d", &newdbg) == 1) {
setdbg = 1;
}
if(cfg_get_by_name(_cfg_corex_ctx, &gname, NULL /* group id */, &vname,
&vval, &vtype)
!= 0) {
rpc->fault(ctx, 500, "Operation failed");
return;
}
olddbg = (int)(long)vval;
if(setdbg == 1) {
cfg_set_now(_cfg_corex_ctx, &gname, NULL /* group id */, &vname,
(void *)(long)newdbg, CFG_VAR_INT);
}
/* add structure node */
if(rpc->add(ctx, "{", &th) < 0) {
rpc->fault(ctx, 500, "Failed creating response");
return;
}
if(setdbg == 1) {
if(rpc->struct_add(th, "dd", "old", olddbg, "new", newdbg) < 0) {
rpc->fault(ctx, 500, "Internal error adding fields");
return;
}
} else {
if(rpc->struct_add(th, "d", "debug", olddbg) < 0) {
rpc->fault(ctx, 500, "Internal error adding fields");
return;
}
}
}
/* clang-format off */
rpc_export_t corex_rpc_cmds[] = {
{"corex.list_sockets", corex_rpc_list_sockets,
corex_rpc_list_sockets_doc, RET_ARRAY},
{"corex.list_aliases", corex_rpc_list_aliases,
corex_rpc_list_aliases_doc, RET_ARRAY},
{"corex.shm_status", corex_rpc_shm_status, corex_rpc_shm_status_doc, 0},
{"corex.shm_summary", corex_rpc_shm_summary, corex_rpc_shm_summary_doc, 0},
{"corex.pkg_summary", corex_rpc_pkg_summary, corex_rpc_pkg_summary_doc, 0},
{"corex.debug", corex_rpc_debug, corex_rpc_debug_doc, 0},
{0, 0, 0, 0}
};
/* clang-format on */
/**
* register RPC commands
*/
int corex_init_rpc(void)
{
if(cfg_register_ctx(&_cfg_corex_ctx, NULL)) {
LOG(L_ERR, "failed to register cfg context\n");
return -1;
}
if(rpc_register_array(corex_rpc_cmds) != 0) {
LM_ERR("failed to register RPC commands\n");
return -1;
}
return 0;
}