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.
325 lines
7.0 KiB
325 lines
7.0 KiB
/*
|
|
* $Id$
|
|
*
|
|
* Copyright (C) 2007 iptelorg GmbH
|
|
*
|
|
* This file is part of ser, a free SIP server.
|
|
*
|
|
* ser is free software; you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License as published by
|
|
* the Free Software Foundation; either version 2 of the License, or
|
|
* (at your option) any later version
|
|
*
|
|
* For a license to use the ser software under conditions
|
|
* other than those described here, or to purchase support for this
|
|
* software, please contact iptel.org by e-mail at the following addresses:
|
|
* info@iptel.org
|
|
*
|
|
* ser is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with this program; if not, write to the Free Software
|
|
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
|
*
|
|
* History
|
|
* -------
|
|
* 2007-12-03 Initial version (Miklos)
|
|
*/
|
|
|
|
#include "../../sr_module.h"
|
|
#include "../../cfg/cfg.h"
|
|
#include "../../cfg/cfg_ctx.h"
|
|
#include "../../rpc.h"
|
|
|
|
static cfg_ctx_t *ctx = NULL;
|
|
|
|
MODULE_VERSION
|
|
|
|
/* module initialization function */
|
|
static int mod_init(void)
|
|
{
|
|
if (cfg_register_ctx(&ctx, NULL)) {
|
|
LOG(L_ERR, "cfg_rpc: failed to register cfg context\n");
|
|
return -1;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static const char* rpc_set_now_doc[2] = {
|
|
"Set the value of a configuration variable and commit the change immediately",
|
|
0
|
|
};
|
|
|
|
static void rpc_set_now_int(rpc_t* rpc, void* c)
|
|
{
|
|
str group, var;
|
|
int i;
|
|
|
|
if (rpc->scan(c, "SSd", &group, &var, &i) < 3)
|
|
return;
|
|
|
|
if (cfg_set_now_int(ctx, &group, &var, i)) {
|
|
rpc->fault(c, 400, "Failed to set the variable");
|
|
return;
|
|
}
|
|
}
|
|
|
|
static void rpc_set_now_string(rpc_t* rpc, void* c)
|
|
{
|
|
str group, var;
|
|
char *ch;
|
|
|
|
if (rpc->scan(c, "SSs", &group, &var, &ch) < 3)
|
|
return;
|
|
|
|
if (cfg_set_now_string(ctx, &group, &var, ch)) {
|
|
rpc->fault(c, 400, "Failed to set the variable");
|
|
return;
|
|
}
|
|
}
|
|
|
|
static const char* rpc_set_delayed_doc[2] = {
|
|
"Prepare the change of a configuration variable, but does not commit the new value yet",
|
|
0
|
|
};
|
|
|
|
static void rpc_set_delayed_int(rpc_t* rpc, void* c)
|
|
{
|
|
str group, var;
|
|
int i;
|
|
|
|
if (rpc->scan(c, "SSd", &group, &var, &i) < 3)
|
|
return;
|
|
|
|
if (cfg_set_delayed_int(ctx, &group, &var, i)) {
|
|
rpc->fault(c, 400, "Failed to set the variable");
|
|
return;
|
|
}
|
|
}
|
|
|
|
static void rpc_set_delayed_string(rpc_t* rpc, void* c)
|
|
{
|
|
str group, var;
|
|
char *ch;
|
|
|
|
if (rpc->scan(c, "SSs", &group, &var, &ch) < 3)
|
|
return;
|
|
|
|
if (cfg_set_delayed_string(ctx, &group, &var, ch)) {
|
|
rpc->fault(c, 400, "Failed to set the variable");
|
|
return;
|
|
}
|
|
}
|
|
|
|
static const char* rpc_commit_doc[2] = {
|
|
"Commit the previously prepared configuration changes",
|
|
0
|
|
};
|
|
|
|
static void rpc_commit(rpc_t* rpc, void* c)
|
|
{
|
|
if (cfg_commit(ctx)) {
|
|
rpc->fault(c, 400, "Failed to commit the changes");
|
|
return;
|
|
}
|
|
}
|
|
|
|
static const char* rpc_rollback_doc[2] = {
|
|
"Drop the prepared configuration changes",
|
|
0
|
|
};
|
|
|
|
static void rpc_rollback(rpc_t* rpc, void* c)
|
|
{
|
|
if (cfg_rollback(ctx)) {
|
|
rpc->fault(c, 400, "Failed to drop the changes");
|
|
return;
|
|
}
|
|
}
|
|
|
|
static const char* rpc_get_doc[2] = {
|
|
"Get the value of a configuration variable",
|
|
0
|
|
};
|
|
|
|
static void rpc_get(rpc_t* rpc, void* c)
|
|
{
|
|
str group, var;
|
|
void *val;
|
|
unsigned int val_type;
|
|
int ret;
|
|
|
|
if (rpc->scan(c, "SS", &group, &var) < 2)
|
|
return;
|
|
|
|
ret = cfg_get_by_name(ctx, &group, &var,
|
|
&val, &val_type);
|
|
if (ret < 0) {
|
|
rpc->fault(c, 400, "Failed to get the variable");
|
|
return;
|
|
|
|
} else if (ret > 0) {
|
|
rpc->fault(c, 400, "Variable exists, but it is not readable via RPC interface");
|
|
return;
|
|
}
|
|
|
|
switch (val_type) {
|
|
case CFG_VAR_INT:
|
|
rpc->add(c, "d", (int)(long)val);
|
|
break;
|
|
|
|
case CFG_VAR_STRING:
|
|
rpc->add(c, "s", (char *)val);
|
|
break;
|
|
|
|
case CFG_VAR_STR:
|
|
rpc->add(c, "S", (str *)val);
|
|
break;
|
|
|
|
case CFG_VAR_POINTER:
|
|
rpc->printf(c, "%p", val);
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
static const char* rpc_help_doc[2] = {
|
|
"Print the description of a configuration variable",
|
|
0
|
|
};
|
|
|
|
static void rpc_help(rpc_t* rpc, void* c)
|
|
{
|
|
str group, var;
|
|
char *ch;
|
|
unsigned int input_type;
|
|
|
|
if (rpc->scan(c, "SS", &group, &var) < 2)
|
|
return;
|
|
|
|
if (cfg_help(ctx, &group, &var,
|
|
&ch, &input_type)
|
|
) {
|
|
rpc->fault(c, 400, "Failed to get the variable description");
|
|
return;
|
|
}
|
|
rpc->add(c, "s", ch);
|
|
|
|
switch (input_type) {
|
|
case CFG_INPUT_INT:
|
|
rpc->printf(c, "(parameter type is integer)");
|
|
break;
|
|
|
|
case CFG_INPUT_STRING:
|
|
case CFG_INPUT_STR:
|
|
rpc->printf(c, "(parameter type is string)");
|
|
break;
|
|
}
|
|
}
|
|
|
|
static const char* rpc_list_doc[2] = {
|
|
"List the configuration variables",
|
|
0
|
|
};
|
|
|
|
static void rpc_list(rpc_t* rpc, void* c)
|
|
{
|
|
void *h;
|
|
str gname;
|
|
cfg_def_t *def;
|
|
int i;
|
|
|
|
cfg_get_group_init(&h);
|
|
while(cfg_get_group_next(&h, &gname, &def))
|
|
for (i=0; def[i].name; i++)
|
|
rpc->printf(c, "%.*s: %s", gname.len, gname.s, def[i].name);
|
|
}
|
|
|
|
static const char* rpc_diff_doc[2] = {
|
|
"List the pending configuration changes that have not been committed yet",
|
|
0
|
|
};
|
|
|
|
static void rpc_diff(rpc_t* rpc, void* c)
|
|
{
|
|
void *h;
|
|
str gname, vname;
|
|
void *old_val, *new_val;
|
|
unsigned int val_type;
|
|
void *rpc_handle;
|
|
|
|
|
|
if (cfg_diff_init(ctx, &h)) {
|
|
rpc->fault(c, 400, "Failed to get the changes");
|
|
return;
|
|
}
|
|
while(cfg_diff_next(&h,
|
|
&gname, &vname,
|
|
&old_val, &new_val,
|
|
&val_type)
|
|
) {
|
|
rpc->add(c, "{", &rpc_handle);
|
|
rpc->struct_add(rpc_handle, "SS",
|
|
"group name", &gname,
|
|
"variable name", &vname);
|
|
|
|
switch (val_type) {
|
|
case CFG_VAR_INT:
|
|
rpc->struct_add(rpc_handle, "dd",
|
|
"old value", (int)(long)old_val,
|
|
"new value", (int)(long)new_val);
|
|
break;
|
|
|
|
case CFG_VAR_STRING:
|
|
rpc->struct_add(rpc_handle, "ss",
|
|
"old value", (char *)old_val,
|
|
"new value", (char *)new_val);
|
|
break;
|
|
|
|
case CFG_VAR_STR:
|
|
rpc->struct_add(rpc_handle, "SS",
|
|
"old value", (str *)old_val,
|
|
"new value", (str *)new_val);
|
|
break;
|
|
|
|
case CFG_VAR_POINTER:
|
|
/* cannot print out pointer value with struct_add() */
|
|
break;
|
|
|
|
}
|
|
}
|
|
cfg_diff_release(ctx);
|
|
}
|
|
|
|
static rpc_export_t rpc_calls[] = {
|
|
{"cfg.set_now_int", rpc_set_now_int, rpc_set_now_doc, 0},
|
|
{"cfg.set_now_string", rpc_set_now_string, rpc_set_now_doc, 0},
|
|
{"cfg.set_delayed_int", rpc_set_delayed_int, rpc_set_delayed_doc, 0},
|
|
{"cfg.set_delayed_string", rpc_set_delayed_string, rpc_set_delayed_doc, 0},
|
|
{"cfg.commit", rpc_commit, rpc_commit_doc, 0},
|
|
{"cfg.rollback", rpc_rollback, rpc_rollback_doc, 0},
|
|
{"cfg.get", rpc_get, rpc_get_doc, 0},
|
|
{"cfg.help", rpc_help, rpc_help_doc, 0},
|
|
{"cfg.list", rpc_list, rpc_list_doc, 0},
|
|
{"cfg.diff", rpc_diff, rpc_diff_doc, 0},
|
|
{0, 0, 0, 0}
|
|
};
|
|
|
|
/* Module interface */
|
|
struct module_exports exports = {
|
|
"cfg_rpc",
|
|
0, /* Exported functions */
|
|
rpc_calls, /* RPC methods */
|
|
0, /* Exported parameters */
|
|
mod_init, /* module initialization function */
|
|
0, /* response function */
|
|
0, /* destroy function */
|
|
0, /* oncancel function */
|
|
0 /* child initialization function */
|
|
};
|