mirror of https://github.com/sipwise/kamailio.git
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.
286 lines
7.1 KiB
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;
|
|
}
|