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/modules/janssonrpc-c/janssonrpc_mod.c

365 lines
8.6 KiB

/**
* Copyright (C) 2013 Flowroute LLC (flowroute.com)
*
* This file is part of Kamailio, a free SIP server.
*
* This file 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
*
*
* This file 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
*
*/
#include <arpa/inet.h>
#include <sys/types.h>
#include <errno.h>
#include "../../pvar.h"
#include "../../mod_fix.h"
#include "../../trim.h"
#include "../../sr_module.h"
#include "../../timer_proc.h"
#include "../../cfg/cfg_struct.h"
#include "../tm/tm_load.h"
#include "../jansson/jansson_utils.h"
#include "janssonrpc_funcs.h"
#include "janssonrpc_request.h"
#include "janssonrpc_io.h"
#include "janssonrpc_connect.h"
#include "janssonrpc_server.h"
#include "janssonrpc_srv.h"
#include "janssonrpc.h"
MODULE_VERSION
static int mod_init(void);
static int child_init(int);
void mod_destroy(void);
int parse_server_param(modparam_t type, void* val);
int parse_retry_codes_param(modparam_t type, void* val);
int parse_min_ttl_param(modparam_t type, void* val);
static int fixup_req(void** param, int param_no);
static int fixup_req_free(void** param, int param_no);
static int fixup_notify(void** param, int param_no);
static int fixup_notify_free(void** param, int param_no);
int fixup_pvar_shm(void** param, int param_no);
int pipe_fds[2] = {-1,-1};
struct tm_binds tmb;
/*
* Exported Functions
*/
int jsonrpc_request_no_options(struct sip_msg* msg,
char* conn,
char* method,
char* params) {
return jsonrpc_request(msg, conn, method, params, NULL);
}
static cmd_export_t cmds[]={
{"janssonrpc_request", (cmd_function)jsonrpc_request,
4, fixup_req, fixup_req_free, ANY_ROUTE},
{"jsansonrpc_request", (cmd_function)jsonrpc_request_no_options,
3, fixup_req, fixup_req_free, ANY_ROUTE},
{"janssonrpc_notification", (cmd_function)jsonrpc_notification,
3, fixup_notify, fixup_notify_free, ANY_ROUTE},
{"mod_janssonrpc_request", (cmd_function)mod_jsonrpc_request,
0, 0, 0, 0},
{0, 0, 0, 0, 0, 0}
};
/*
* Script Parameters
*/
static param_export_t mod_params[]={
{"server", STR_PARAM|USE_FUNC_PARAM, (void*)parse_server_param},
{"retry_codes", STR_PARAM|USE_FUNC_PARAM, (void*)parse_retry_codes_param},
{"min_srv_ttl", INT_PARAM|USE_FUNC_PARAM, (void*)parse_min_ttl_param},
{"result_pv", STR_PARAM, &result_pv_str.s},
{ 0,0,0 }
};
/*
* Exports
*/
struct module_exports exports = {
"janssonrpc-c", /* module name */
DEFAULT_DLFLAGS, /* dlopen flags */
cmds, /* Exported functions */
mod_params, /* Exported parameters */
0, /* exported statistics */
0, /* exported MI functions */
0, /* exported pseudo-variables */
0, /* extra processes */
mod_init, /* module initialization function */
0, /* response function*/
mod_destroy, /* destroy function */
child_init /* per-child init function */
};
static int mod_init(void)
{
/* load the tm functions */
if(load_tm_api(&tmb)<0) return -1;
/* load json_to_val from json module */
jsontoval = (jansson_to_val_f)find_export("jansson_to_val", 0, 0);
if(jsontoval == 0) {
ERR("ERROR:jsonrpc:mod_init: cannot import json_to_val\n");
return -1;
}
/* setup result pvar */
if (result_pv_str.s == NULL)
result_pv_str.s = JSONRPC_RESULT_STR;
result_pv_str.len = strlen(result_pv_str.s);
if(pv_parse_spec(&result_pv_str, &jsonrpc_result_pv)<0) {
ERR("cannot parse result_pv: %.*s\n", STR(result_pv_str));
return -1;
}
if(!(pv_is_w(&jsonrpc_result_pv))) {
ERR("%.*s is read only\n", STR(result_pv_str));
return -1;
}
register_procs(1);
register_basic_timers(1);
if (pipe(pipe_fds) < 0) {
ERR("pipe() failed\n");
return -1;
}
if(jsonrpc_min_srv_ttl < ABSOLUTE_MIN_SRV_TTL) {
jsonrpc_min_srv_ttl = JSONRPC_DEFAULT_MIN_SRV_TTL; /* 5s */
}
return 0;
}
static int child_init(int rank)
{
int pid;
if (rank!=PROC_INIT)
cmd_pipe = pipe_fds[1];
if (rank!=PROC_MAIN)
return 0;
jsonrpc_server_group_lock = lock_alloc();
if(jsonrpc_server_group_lock == NULL) {
ERR("cannot allocate the server_group_lock\n");
return -1;
}
if(lock_init(jsonrpc_server_group_lock) == 0) {
ERR("failed to initialized the server_group_lock\n");
lock_dealloc(jsonrpc_server_group_lock);
return -1;
}
srv_cb_params_t* params = (srv_cb_params_t*)shm_malloc(sizeof(srv_cb_params_t));
CHECK_MALLOC(params);
params->cmd_pipe = pipe_fds[1];
params->srv_ttl = jsonrpc_min_srv_ttl;
/* start timer to check SRV ttl every second */
if(fork_basic_timer(PROC_TIMER, "jsonrpc SRV timer", 1 /*socks flag*/,
refresh_srv_cb, (void*)params, ABSOLUTE_MIN_SRV_TTL)<0) {
ERR("Failed to start SRV timer\n");
return -1;
}
pid=fork_process(PROC_RPC, "jsonrpc io handler", 1);
if (pid<0)
return -1; /* error */
if(pid==0){
/* child */
if (cfg_child_init()) return -1;
close(pipe_fds[1]);
return jsonrpc_io_child_process(pipe_fds[0]);
}
return 0;
}
void mod_destroy(void)
{
lock_get(jsonrpc_server_group_lock); /* blocking */
if(jsonrpc_server_group_lock) lock_dealloc(jsonrpc_server_group_lock);
free_server_group(global_server_group);
CHECK_AND_FREE(global_server_group);
}
int parse_server_param(modparam_t type, void* val)
{
if(global_server_group == NULL) {
global_server_group = shm_malloc(sizeof(void*));
*global_server_group = NULL;
}
return jsonrpc_parse_server((char*)val, global_server_group);
}
/* helper function for parse_retry_codes_param */
int s2i(char* str, int* result)
{
char* endptr;
errno = 0;
long val = strtol(str, &endptr, 10);
if ((errno == ERANGE && (val == LONG_MAX || val == LONG_MIN))
|| (errno != 0 && val == 0)) {
ERR("%s is not a number: %s\n", str, strerror(errno));
return -1;
}
if (endptr == str) {
ERR("failed to convert %s to integer\n", str);
return -1;
}
*result = (int)val;
return 0;
}
int parse_retry_codes_param(modparam_t type, void* val)
{
if (val==NULL) {
ERR("retry_codes cannot be NULL!\n");
return -1;
}
if (PARAM_TYPE_MASK(type) != STR_PARAM) {
ERR("retry_codes must be a string\n");
return -1;
}
global_retry_ranges = NULL;
char* save_comma;
char* save_elipse;
char* token;
char* start_s;
int start;
char* end_s;
int end;
char* codes_s = (char*)val;
char* tmp;
retry_range_t** tmp_range;
tmp_range = &global_retry_ranges;
for (tmp = codes_s; ; tmp = NULL) {
token = strtok_r(tmp, ",", &save_comma);
if (token == NULL)
break;
start_s = strtok_r(token, ".", &save_elipse);
if (start_s == NULL) {
continue;
}
if(s2i(start_s, &start)<0) return -1;
*tmp_range = shm_malloc(sizeof(retry_range_t));
CHECK_MALLOC(*tmp_range);
memset(*tmp_range, 0, sizeof(retry_range_t));
(*tmp_range)->start = start;
end_s = strtok_r(NULL, ".", &save_elipse);
if (end_s == NULL) {
end_s = start_s;
}
if(s2i(end_s, &end)<0) return -1;
(*tmp_range)->end = end;
tmp_range = &((*tmp_range)->next);
}
return 0;
}
int parse_min_ttl_param(modparam_t type, void* val)
{
if (val==0) {
ERR("min_srv_ttl cannot be NULL!\n");
return -1;
}
if (PARAM_TYPE_MASK(type) != INT_PARAM) {
ERR("min_srv_ttl must be of type %d, not %d!\n", INT_PARAM, type);
return -1;
}
jsonrpc_min_srv_ttl = (int)(long)val;
if(jsonrpc_min_srv_ttl < ABSOLUTE_MIN_SRV_TTL) {
ERR("Cannot set min_srv_ttl lower than %d", ABSOLUTE_MIN_SRV_TTL);
return -1;
}
INFO("min_srv_ttl set to %d\n", jsonrpc_min_srv_ttl);
return 0;
}
/* Fixup Functions */
static int fixup_req(void** param, int param_no)
{
if (param_no <= 4) {
return fixup_spve_null(param, 1);
}
ERR("function takes at most 4 parameters.\n");
return -1;
}
static int fixup_req_free(void** param, int param_no)
{
if (param_no <= 4) {
return fixup_free_spve_null(param, 1);
}
ERR("function takes at most 4 parameters.\n");
return -1;
}
static int fixup_notify(void** param, int param_no)
{
if (param_no <= 3) {
return fixup_spve_null(param, 1);
}
ERR("function takes at most 3 parameters.\n");
return -1;
}
static int fixup_notify_free(void** param, int param_no)
{
if (param_no <= 3) {
return fixup_free_spve_null(param, 1);
}
ERR("function takes at most 3 parameters.\n");
return -1;
}