mirror of
https://github.com/signalwire/freeswitch.git
synced 2025-03-05 18:13:27 +00:00
add mod_cdr_csv
git-svn-id: http://svn.freeswitch.org/svn/freeswitch/trunk@6542 d0543943-73ff-0310-b7d9-9358b9ac24b2
This commit is contained in:
parent
773894bc2d
commit
8bf63d3d8c
@ -40,6 +40,7 @@ endpoints/mod_sofia
|
|||||||
#../../libs/openzap/mod_openzap
|
#../../libs/openzap/mod_openzap
|
||||||
#event_handlers/mod_event_multicast
|
#event_handlers/mod_event_multicast
|
||||||
event_handlers/mod_event_socket
|
event_handlers/mod_event_socket
|
||||||
|
event_handlers/mod_cdr_csv
|
||||||
#event_handlers/mod_radius_cdr
|
#event_handlers/mod_radius_cdr
|
||||||
formats/mod_native_file
|
formats/mod_native_file
|
||||||
formats/mod_sndfile
|
formats/mod_sndfile
|
||||||
|
12
conf/autoload_configs/cdr_csv.conf.xml
Normal file
12
conf/autoload_configs/cdr_csv.conf.xml
Normal file
@ -0,0 +1,12 @@
|
|||||||
|
<configuration name="cdr_csv.conf" description="CDR CSV Format">
|
||||||
|
<settings>
|
||||||
|
<!-- 'cdr-csv' will always be appended to log-base -->
|
||||||
|
<!--<param name="log-base" value="/var/log"/>-->
|
||||||
|
<param name="default-template" value="example"/>
|
||||||
|
<param name="rotate-on-hup" value="true"/>
|
||||||
|
</settings>
|
||||||
|
<templates>
|
||||||
|
<template name="example">"${caller_id_name}","${caller_id_number}","${destination_number}","${context}","${start_stamp}","${answer_stamp}","${end_stamp}","${duration}","${billsec}","${hangup_cause}","${uuid}","${bleg_uuid}", "${accountcode}"</template>
|
||||||
|
<template name="asterisk">"${accountcode}","${caller_id_number}","${destination_number}","${context}","${caller_id}","${channel_name}","${bridge_channel}","${last_app}","${last_arg}","${start_stamp}","${answer_stamp}","${end_stamp}","${duration}","${billsec}","${hangup_cause}","${amaflags}","${uuid}","${userfield}"</template>
|
||||||
|
</templates>
|
||||||
|
</configuration>
|
@ -15,7 +15,7 @@
|
|||||||
<!-- <load module="mod_xml_cdr"/> -->
|
<!-- <load module="mod_xml_cdr"/> -->
|
||||||
|
|
||||||
<!-- Event Handlers -->
|
<!-- Event Handlers -->
|
||||||
<!-- <load module="mod_cdr"/> -->
|
<load module="mod_cdr_csv"/>
|
||||||
<!-- <load module="mod_event_multicast"/> -->
|
<!-- <load module="mod_event_multicast"/> -->
|
||||||
<!-- <load module="mod_event_socket"/> -->
|
<!-- <load module="mod_event_socket"/> -->
|
||||||
<!-- <load module="mod_xmpp_event"/> -->
|
<!-- <load module="mod_xmpp_event"/> -->
|
||||||
|
@ -91,6 +91,10 @@ SWITCH_BEGIN_EXTERN_C
|
|||||||
#define SWITCH_SEQ_CLEARLINE SWITCH_SEQ_ESC SWITCH_SEQ_CLEARLINE_CHAR_STR
|
#define SWITCH_SEQ_CLEARLINE SWITCH_SEQ_ESC SWITCH_SEQ_CLEARLINE_CHAR_STR
|
||||||
#define SWITCH_SEQ_CLEARLINEEND SWITCH_SEQ_ESC SWITCH_SEQ_CLEARLINEEND_CHAR
|
#define SWITCH_SEQ_CLEARLINEEND SWITCH_SEQ_ESC SWITCH_SEQ_CLEARLINEEND_CHAR
|
||||||
#define SWITCH_SEQ_CLEARSCR SWITCH_SEQ_ESC SWITCH_SEQ_CLEARSCR_CHAR SWITCH_SEQ_HOME
|
#define SWITCH_SEQ_CLEARSCR SWITCH_SEQ_ESC SWITCH_SEQ_CLEARSCR_CHAR SWITCH_SEQ_HOME
|
||||||
|
|
||||||
|
|
||||||
|
#define SWITCH_DEFAULT_DIR_PERMS SWITCH_FPROT_UREAD | SWITCH_FPROT_UWRITE | SWITCH_FPROT_UEXECUTE | SWITCH_FPROT_GREAD | SWITCH_FPROT_GEXECUTE
|
||||||
|
|
||||||
#ifdef WIN32
|
#ifdef WIN32
|
||||||
#define SWITCH_PATH_SEPARATOR "\\"
|
#define SWITCH_PATH_SEPARATOR "\\"
|
||||||
#else
|
#else
|
||||||
@ -98,6 +102,7 @@ SWITCH_BEGIN_EXTERN_C
|
|||||||
#endif
|
#endif
|
||||||
#define SWITCH_URL_SEPARATOR "://"
|
#define SWITCH_URL_SEPARATOR "://"
|
||||||
#define SWITCH_BRIDGE_CHANNEL_VARIABLE "bridge_channel"
|
#define SWITCH_BRIDGE_CHANNEL_VARIABLE "bridge_channel"
|
||||||
|
#define SWITCH_CHANNEL_NAME_VARIABLE "channel_name"
|
||||||
#define SWITCH_BRIDGE_UUID_VARIABLE "bridge_uuid"
|
#define SWITCH_BRIDGE_UUID_VARIABLE "bridge_uuid"
|
||||||
#define SWITCH_PLAYBACK_TERMINATORS_VARIABLE "playback_terminators"
|
#define SWITCH_PLAYBACK_TERMINATORS_VARIABLE "playback_terminators"
|
||||||
#define SWITCH_CACHE_SPEECH_HANDLES_VARIABLE "cache_speech_handles"
|
#define SWITCH_CACHE_SPEECH_HANDLES_VARIABLE "cache_speech_handles"
|
||||||
|
@ -875,8 +875,6 @@ typedef enum {
|
|||||||
MSG_SAVED
|
MSG_SAVED
|
||||||
} msg_type_t;
|
} msg_type_t;
|
||||||
|
|
||||||
static uint32_t DEFAULT_DIR_PERMS = SWITCH_FPROT_UREAD | SWITCH_FPROT_UWRITE | SWITCH_FPROT_UEXECUTE | SWITCH_FPROT_GREAD | SWITCH_FPROT_GEXECUTE;
|
|
||||||
|
|
||||||
|
|
||||||
static switch_status_t create_file(switch_core_session_t *session, vm_profile_t *profile, char *macro_name, char *file_path, switch_size_t *message_len)
|
static switch_status_t create_file(switch_core_session_t *session, vm_profile_t *profile, char *macro_name, char *file_path, switch_size_t *message_len)
|
||||||
{
|
{
|
||||||
@ -1640,7 +1638,7 @@ static void voicemail_check_main(switch_core_session_t *session, const char *pro
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
if (switch_dir_make_recursive(dir_path, DEFAULT_DIR_PERMS, switch_core_session_get_pool(session)) != SWITCH_STATUS_SUCCESS) {
|
if (switch_dir_make_recursive(dir_path, SWITCH_DEFAULT_DIR_PERMS, switch_core_session_get_pool(session)) != SWITCH_STATUS_SUCCESS) {
|
||||||
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_ERROR, "Error creating %s\n", dir_path);
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_ERROR, "Error creating %s\n", dir_path);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
@ -1739,7 +1737,7 @@ static switch_status_t voicemail_leave_main(switch_core_session_t *session, cons
|
|||||||
id);
|
id);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (switch_dir_make_recursive(dir_path, DEFAULT_DIR_PERMS, switch_core_session_get_pool(session)) != SWITCH_STATUS_SUCCESS) {
|
if (switch_dir_make_recursive(dir_path, SWITCH_DEFAULT_DIR_PERMS, switch_core_session_get_pool(session)) != SWITCH_STATUS_SUCCESS) {
|
||||||
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_ERROR, "Error creating %s\n", dir_path);
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_ERROR, "Error creating %s\n", dir_path);
|
||||||
goto end;
|
goto end;
|
||||||
}
|
}
|
||||||
@ -2069,7 +2067,7 @@ SWITCH_STANDARD_APP(voicemail_function)
|
|||||||
channel = switch_core_session_get_channel(session);
|
channel = switch_core_session_get_channel(session);
|
||||||
assert(channel != NULL);
|
assert(channel != NULL);
|
||||||
|
|
||||||
if (switch_dir_make_recursive(SWITCH_GLOBAL_dirs.storage_dir, DEFAULT_DIR_PERMS, switch_core_session_get_pool(session)) != SWITCH_STATUS_SUCCESS) {
|
if (switch_dir_make_recursive(SWITCH_GLOBAL_dirs.storage_dir, SWITCH_DEFAULT_DIR_PERMS, switch_core_session_get_pool(session)) != SWITCH_STATUS_SUCCESS) {
|
||||||
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_ERROR, "Error creating %s\n", SWITCH_GLOBAL_dirs.storage_dir);
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_ERROR, "Error creating %s\n", SWITCH_GLOBAL_dirs.storage_dir);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
1
src/mod/event_handlers/mod_cdr_csv/Makefile
Normal file
1
src/mod/event_handlers/mod_cdr_csv/Makefile
Normal file
@ -0,0 +1 @@
|
|||||||
|
include ../../../../build/modmake.rules
|
435
src/mod/event_handlers/mod_cdr_csv/mod_cdr_csv.c
Normal file
435
src/mod/event_handlers/mod_cdr_csv/mod_cdr_csv.c
Normal file
@ -0,0 +1,435 @@
|
|||||||
|
/*
|
||||||
|
* FreeSWITCH Modular Media Switching Software Library / Soft-Switch Application
|
||||||
|
* Copyright (C) 2005/2006, Anthony Minessale II <anthmct@yahoo.com>
|
||||||
|
*
|
||||||
|
* Version: MPL 1.1
|
||||||
|
*
|
||||||
|
* The contents of this file are subject to the Mozilla Public License Version
|
||||||
|
* 1.1 (the "License"); you may not use this file except in compliance with
|
||||||
|
* the License. You may obtain a copy of the License at
|
||||||
|
* http://www.mozilla.org/MPL/
|
||||||
|
*
|
||||||
|
* Software distributed under the License is distributed on an "AS IS" basis,
|
||||||
|
* WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License
|
||||||
|
* for the specific language governing rights and limitations under the
|
||||||
|
* License.
|
||||||
|
*
|
||||||
|
* The Original Code is FreeSWITCH Modular Media Switching Software Library / Soft-Switch Application
|
||||||
|
*
|
||||||
|
* The Initial Developer of the Original Code is
|
||||||
|
* Anthony Minessale II <anthmct@yahoo.com>
|
||||||
|
* Portions created by the Initial Developer are Copyright (C)
|
||||||
|
* the Initial Developer. All Rights Reserved.
|
||||||
|
*
|
||||||
|
* Contributor(s):
|
||||||
|
*
|
||||||
|
* mod_cdr_csv.c -- Asterisk Compatible CDR Module
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
#include <sys/stat.h>
|
||||||
|
#include <switch.h>
|
||||||
|
|
||||||
|
struct cdr_fd {
|
||||||
|
int fd;
|
||||||
|
char *path;
|
||||||
|
int64_t bytes;
|
||||||
|
switch_mutex_t *mutex;
|
||||||
|
};
|
||||||
|
typedef struct cdr_fd cdr_fd_t;
|
||||||
|
|
||||||
|
const char *default_template =
|
||||||
|
"\"${caller_id_name}\",\"${caller_id_number}\",\"${destination_number}\",\"${context}\",\"${start_stamp}\","
|
||||||
|
"\"${answer_stamp}\",\"${end_stamp}\",\"${duration}\",\"${billsec}\",\"${hangup_cause}\",\"${uuid}\",\"${bleg_uuid}\", \"${accountcode}\"";
|
||||||
|
|
||||||
|
static struct {
|
||||||
|
switch_memory_pool_t *pool;
|
||||||
|
switch_hash_t *fd_hash;
|
||||||
|
switch_hash_t *template_hash;
|
||||||
|
char *log_dir;
|
||||||
|
char *default_template;
|
||||||
|
int shutdown;
|
||||||
|
int rotate;
|
||||||
|
} globals;
|
||||||
|
|
||||||
|
SWITCH_MODULE_LOAD_FUNCTION(mod_cdr_csv_load);
|
||||||
|
SWITCH_MODULE_SHUTDOWN_FUNCTION(mod_cdr_csv_shutdown);
|
||||||
|
SWITCH_MODULE_DEFINITION(mod_cdr_csv, mod_cdr_csv_load, NULL, NULL);
|
||||||
|
|
||||||
|
static off_t fd_size(int fd)
|
||||||
|
{
|
||||||
|
struct stat s = {0};
|
||||||
|
fstat(fd, &s);
|
||||||
|
return s.st_size;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void do_reopen(cdr_fd_t *fd)
|
||||||
|
{
|
||||||
|
|
||||||
|
if (fd->fd > -1) {
|
||||||
|
close(fd->fd);
|
||||||
|
fd->fd = -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
if ((fd->fd = open(fd->path, O_WRONLY | O_CREAT | O_APPEND, S_IRUSR | S_IWUSR)) > -1) {
|
||||||
|
fd->bytes = fd_size(fd->fd);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static void do_rotate(cdr_fd_t *fd)
|
||||||
|
{
|
||||||
|
switch_time_exp_t tm;
|
||||||
|
char date[80] = "";
|
||||||
|
switch_size_t retsize;
|
||||||
|
char *p;
|
||||||
|
size_t len;
|
||||||
|
|
||||||
|
close(fd->fd);
|
||||||
|
|
||||||
|
if (globals.rotate) {
|
||||||
|
switch_time_exp_lt(&tm, switch_time_now());
|
||||||
|
switch_strftime(date, &retsize, sizeof(date), "%Y-%m-%d-%H-%M-%S", &tm);
|
||||||
|
|
||||||
|
len = strlen(fd->path) + strlen(date) + 2;
|
||||||
|
p = switch_mprintf("%s.%s", fd->path, date);
|
||||||
|
assert(p);
|
||||||
|
switch_file_rename(fd->path, p, globals.pool);
|
||||||
|
free(p);
|
||||||
|
}
|
||||||
|
|
||||||
|
do_reopen(fd);
|
||||||
|
|
||||||
|
if (fd->fd < 0) {
|
||||||
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_ERROR, "Error opening %s\n", fd->path);
|
||||||
|
} else {
|
||||||
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_NOTICE, "%s CDR logfile %s\n", globals.rotate ? "Rotated" : "Re-opened", fd->path);
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
static void write_cdr(const char *path, const char *log_line)
|
||||||
|
{
|
||||||
|
cdr_fd_t *fd = NULL;
|
||||||
|
|
||||||
|
if ((fd = switch_core_hash_find(globals.fd_hash, path))) {
|
||||||
|
switch_mutex_lock(fd->mutex);
|
||||||
|
} else {
|
||||||
|
int lfd = open(path, O_WRONLY | O_CREAT | O_APPEND, S_IRUSR | S_IWUSR);
|
||||||
|
|
||||||
|
if (lfd > -1) {
|
||||||
|
fd = switch_core_alloc(globals.pool, sizeof(*fd));
|
||||||
|
switch_mutex_init(&fd->mutex, SWITCH_MUTEX_NESTED, globals.pool);
|
||||||
|
fd->fd = lfd;
|
||||||
|
switch_mutex_lock(fd->mutex);
|
||||||
|
fd->path = switch_core_strdup(globals.pool, path);
|
||||||
|
fd->bytes = fd_size(fd->fd);
|
||||||
|
switch_core_hash_insert(globals.fd_hash, path, fd);
|
||||||
|
} else {
|
||||||
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_ERROR, "Error opening %s\n", path);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (fd) {
|
||||||
|
size_t bytes_in, bytes_out;
|
||||||
|
bytes_out = strlen(log_line);
|
||||||
|
|
||||||
|
if (fd->bytes + bytes_out > UINT_MAX) {
|
||||||
|
do_rotate(fd);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (fd->fd < 0) {
|
||||||
|
do_reopen(fd);
|
||||||
|
if (fd->fd < 0) {
|
||||||
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_ERROR, "Error opening %s\n", path);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if ((bytes_in = write(fd->fd, log_line, bytes_out)) != bytes_out) {
|
||||||
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_CRIT, "Write error to file %s\n", path);
|
||||||
|
}
|
||||||
|
fd->bytes += bytes_in;
|
||||||
|
switch_mutex_unlock(fd->mutex);
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
static switch_status_t my_on_hangup(switch_core_session_t *session)
|
||||||
|
{
|
||||||
|
switch_channel_t *channel = switch_core_session_get_channel(session);
|
||||||
|
switch_status_t status = SWITCH_STATUS_FALSE;
|
||||||
|
const char *log_dir = NULL, *accountcode = NULL, *cid_buf = NULL, *template = NULL, *template_str = NULL;
|
||||||
|
char *log_line, *path = NULL;
|
||||||
|
switch_caller_profile_t *caller_profile, *ocp;
|
||||||
|
|
||||||
|
switch_app_log_t *app_log, *ap;
|
||||||
|
char *last_app = NULL, *last_arg = NULL;
|
||||||
|
char start[80] = "", answer[80] = "", end[80] = "", tmp[30] = "";
|
||||||
|
int32_t duration = 0, billsec = 0;
|
||||||
|
|
||||||
|
if (switch_channel_get_originator_caller_profile(channel)) {
|
||||||
|
return SWITCH_STATUS_SUCCESS;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!(log_dir = switch_channel_get_variable(channel, "cdr_csv_base"))) {
|
||||||
|
log_dir = globals.log_dir;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (switch_dir_make_recursive(log_dir, SWITCH_DEFAULT_DIR_PERMS, switch_core_session_get_pool(session)) != SWITCH_STATUS_SUCCESS) {
|
||||||
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_ERROR, "Error creating %s\n", log_dir);
|
||||||
|
return SWITCH_STATUS_FALSE;
|
||||||
|
}
|
||||||
|
|
||||||
|
caller_profile = switch_channel_get_caller_profile(channel);
|
||||||
|
ocp = switch_channel_get_originatee_caller_profile(channel);
|
||||||
|
|
||||||
|
if (!switch_strlen_zero(caller_profile->caller_id_name)) {
|
||||||
|
cid_buf = switch_core_session_sprintf(session, "\"%s\" <%s>", caller_profile->caller_id_name,
|
||||||
|
switch_str_nil(caller_profile->caller_id_number));
|
||||||
|
} else {
|
||||||
|
cid_buf = caller_profile->caller_id_number;
|
||||||
|
}
|
||||||
|
|
||||||
|
if ((app_log = switch_core_session_get_app_log(session))) {
|
||||||
|
for (ap = app_log; ap && ap->next; ap = ap->next);
|
||||||
|
last_app = ap->app;
|
||||||
|
last_arg = ap->arg;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (caller_profile->times) {
|
||||||
|
switch_time_exp_t tm;
|
||||||
|
switch_size_t retsize;
|
||||||
|
const char *fmt = "%Y-%m-%d %T";
|
||||||
|
|
||||||
|
switch_time_exp_lt(&tm, caller_profile->times->created);
|
||||||
|
switch_strftime(start, &retsize, sizeof(start), fmt, &tm);
|
||||||
|
|
||||||
|
switch_time_exp_lt(&tm, caller_profile->times->answered);
|
||||||
|
switch_strftime(answer, &retsize, sizeof(answer), fmt, &tm);
|
||||||
|
|
||||||
|
switch_time_exp_lt(&tm, caller_profile->times->hungup);
|
||||||
|
switch_strftime(end, &retsize, sizeof(end), fmt, &tm);
|
||||||
|
|
||||||
|
duration = ((int32_t) caller_profile->times->hungup / 1000000) - ((int32_t) caller_profile->times->created / 1000000);
|
||||||
|
if (duration < 0) {
|
||||||
|
duration = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
billsec = ((int32_t) caller_profile->times->hungup / 1000000) - ((int32_t) caller_profile->times->answered / 1000000);
|
||||||
|
if (billsec < 0) {
|
||||||
|
billsec = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
switch_channel_set_variable(channel, "start_stamp", start);
|
||||||
|
switch_channel_set_variable(channel, "answer_stamp", answer);
|
||||||
|
switch_channel_set_variable(channel, "end_stamp", end);
|
||||||
|
switch_channel_set_variable(channel, "last_app", last_app);
|
||||||
|
switch_channel_set_variable(channel, "last_arg", last_arg);
|
||||||
|
switch_channel_set_variable(channel, "caller_id", cid_buf);
|
||||||
|
|
||||||
|
snprintf(tmp, sizeof(tmp), "%d", duration);
|
||||||
|
switch_channel_set_variable(channel, "duration", tmp);
|
||||||
|
|
||||||
|
snprintf(tmp, sizeof(tmp), "%d", billsec);
|
||||||
|
switch_channel_set_variable(channel, "billsec", tmp);
|
||||||
|
|
||||||
|
|
||||||
|
if (!(template = switch_channel_get_variable(channel, "cdr_template"))) {
|
||||||
|
template = globals.default_template;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!(template_str = (const char *) switch_core_hash_find(globals.template_hash, template))) {
|
||||||
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_ERROR, "Error finding template %s substituting default\n", template);
|
||||||
|
template_str = (const char *) switch_core_hash_find(globals.template_hash, globals.default_template);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
log_line = switch_channel_expand_variables(channel, template_str);
|
||||||
|
#if 0
|
||||||
|
log_line = switch_core_session_sprintf(session,
|
||||||
|
"\"%s\",\"%s\",\"%s\",\"%s\",\"%s\",\"%s\",\"%s\",\"%s\",\"%s\",\"%s\","
|
||||||
|
"\"%s\",\"%s\",\"%d\",\"%d\",\"%s\",\"%s\",\"%s\",\"%s\"\n",
|
||||||
|
switch_str_nil(accountcode),
|
||||||
|
switch_str_nil(caller_profile->caller_id_number),
|
||||||
|
switch_str_nil(caller_profile->destination_number),
|
||||||
|
switch_str_nil(caller_profile->context),
|
||||||
|
cid_buf,
|
||||||
|
switch_channel_get_name(channel),
|
||||||
|
switch_str_nil(switch_channel_get_variable(channel, SWITCH_BRIDGE_CHANNEL_VARIABLE)),
|
||||||
|
switch_str_nil(last_app),
|
||||||
|
switch_str_nil(last_arg),
|
||||||
|
start,
|
||||||
|
answer,
|
||||||
|
end,
|
||||||
|
duration,
|
||||||
|
billsec,
|
||||||
|
switch_str_nil(switch_channel_cause2str(switch_channel_get_cause(channel))),
|
||||||
|
switch_str_nil(switch_channel_get_variable(channel, "AMAFLAGS")),
|
||||||
|
switch_str_nil(caller_profile->uuid),
|
||||||
|
switch_str_nil(switch_channel_get_variable(channel, "USERFIELD"))
|
||||||
|
);
|
||||||
|
|
||||||
|
#endif
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
if ((accountcode = switch_channel_get_variable(channel, "ACCOUNTCODE"))) {
|
||||||
|
path = switch_mprintf("%s%s%s.csv", log_dir, SWITCH_PATH_SEPARATOR, accountcode);
|
||||||
|
assert(path);
|
||||||
|
write_cdr(path, log_line);
|
||||||
|
free(path);
|
||||||
|
}
|
||||||
|
|
||||||
|
path = switch_mprintf("%s%sMaster.csv", log_dir, SWITCH_PATH_SEPARATOR);
|
||||||
|
assert(path);
|
||||||
|
write_cdr(path, log_line);
|
||||||
|
free(path);
|
||||||
|
|
||||||
|
|
||||||
|
if (log_line && log_line != template_str) {
|
||||||
|
free(log_line);
|
||||||
|
}
|
||||||
|
|
||||||
|
return status;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static void event_handler(switch_event_t *event)
|
||||||
|
{
|
||||||
|
const char *sig = switch_event_get_header(event, "Trapped-Signal");
|
||||||
|
switch_hash_index_t *hi;
|
||||||
|
void *val;
|
||||||
|
cdr_fd_t *fd;
|
||||||
|
|
||||||
|
if (sig && !strcmp(sig, "HUP")) {
|
||||||
|
for (hi = switch_hash_first(NULL, globals.fd_hash); hi; hi = switch_hash_next(hi)) {
|
||||||
|
switch_hash_this(hi, NULL, NULL, &val);
|
||||||
|
fd = (cdr_fd_t *) val;
|
||||||
|
do_rotate(fd);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static switch_state_handler_table_t state_handlers = {
|
||||||
|
/*.on_init */ NULL,
|
||||||
|
/*.on_ring */ NULL,
|
||||||
|
/*.on_execute */ NULL,
|
||||||
|
/*.on_hangup */ my_on_hangup,
|
||||||
|
/*.on_loopback */ NULL,
|
||||||
|
/*.on_transmit */ NULL
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
static switch_status_t load_config(switch_memory_pool_t *pool)
|
||||||
|
{
|
||||||
|
char *cf = "cdr_csv.conf";
|
||||||
|
switch_xml_t cfg, xml, settings, param;
|
||||||
|
switch_status_t status = SWITCH_STATUS_SUCCESS;
|
||||||
|
|
||||||
|
memset(&globals,0,sizeof(globals));
|
||||||
|
switch_core_hash_init(&globals.fd_hash, pool);
|
||||||
|
switch_core_hash_init(&globals.template_hash, pool);
|
||||||
|
|
||||||
|
globals.pool = pool;
|
||||||
|
|
||||||
|
switch_core_hash_insert(globals.template_hash, "default", default_template);
|
||||||
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_DEBUG, "Adding default template.\n");
|
||||||
|
|
||||||
|
if ((xml = switch_xml_open_cfg(cf, &cfg, NULL))) {
|
||||||
|
|
||||||
|
if ((settings = switch_xml_child(cfg, "settings"))) {
|
||||||
|
for (param = switch_xml_child(settings, "param"); param; param = param->next) {
|
||||||
|
char *var = (char *) switch_xml_attr_soft(param, "name");
|
||||||
|
char *val = (char *) switch_xml_attr_soft(param, "value");
|
||||||
|
|
||||||
|
if (!strcasecmp(var, "log-base")) {
|
||||||
|
globals.log_dir = switch_core_sprintf(pool, "%s%scdr-csv", val, SWITCH_PATH_SEPARATOR);
|
||||||
|
} else if (!strcasecmp(var, "rotate-on-hup")) {
|
||||||
|
globals.rotate = switch_true(val);
|
||||||
|
} else if (!strcasecmp(var, "default-template")) {
|
||||||
|
globals.default_template = switch_core_strdup(pool, val);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if ((settings = switch_xml_child(cfg, "templates"))) {
|
||||||
|
for (param = switch_xml_child(settings, "template"); param; param = param->next) {
|
||||||
|
char *var = (char *) switch_xml_attr(param, "name");
|
||||||
|
if (var) {
|
||||||
|
char *tpl;
|
||||||
|
size_t len = strlen(param->txt) + 2;
|
||||||
|
if (end_of(param->txt) != '\n') {
|
||||||
|
tpl = switch_core_alloc(pool, len);
|
||||||
|
snprintf(tpl, len, "%s\n", param->txt);
|
||||||
|
} else {
|
||||||
|
tpl = switch_core_strdup(pool, param->txt);
|
||||||
|
}
|
||||||
|
|
||||||
|
switch_core_hash_insert(globals.template_hash, var, tpl);
|
||||||
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_DEBUG, "Adding template %s.\n", var);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
switch_xml_free(xml);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
if (switch_strlen_zero(globals.default_template)) {
|
||||||
|
globals.default_template = switch_core_strdup(pool, "default");
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!globals.log_dir) {
|
||||||
|
globals.log_dir = switch_core_sprintf(pool, "%s%scdr-csv", SWITCH_GLOBAL_dirs.log_dir, SWITCH_PATH_SEPARATOR);
|
||||||
|
}
|
||||||
|
|
||||||
|
return status;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
SWITCH_MODULE_LOAD_FUNCTION(mod_cdr_csv_load)
|
||||||
|
{
|
||||||
|
switch_status_t status = SWITCH_STATUS_SUCCESS;
|
||||||
|
|
||||||
|
if (switch_event_bind((char *) modname, SWITCH_EVENT_TRAP, SWITCH_EVENT_SUBCLASS_ANY, event_handler, NULL) != SWITCH_STATUS_SUCCESS) {
|
||||||
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_ERROR, "Couldn't bind!\n");
|
||||||
|
return SWITCH_STATUS_GENERR;
|
||||||
|
}
|
||||||
|
|
||||||
|
switch_core_add_state_handler(&state_handlers);
|
||||||
|
*module_interface = switch_loadable_module_create_module_interface(pool, modname);
|
||||||
|
|
||||||
|
load_config(pool);
|
||||||
|
|
||||||
|
if ((status = switch_dir_make_recursive(globals.log_dir, SWITCH_DEFAULT_DIR_PERMS, pool)) != SWITCH_STATUS_SUCCESS) {
|
||||||
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_ERROR, "Error creating %s\n", globals.log_dir);
|
||||||
|
}
|
||||||
|
|
||||||
|
return status;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
SWITCH_MODULE_SHUTDOWN_FUNCTION(mod_cdr_csv_shutdown)
|
||||||
|
{
|
||||||
|
|
||||||
|
globals.shutdown=1;
|
||||||
|
return SWITCH_STATUS_SUCCESS;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
/* For Emacs:
|
||||||
|
* Local Variables:
|
||||||
|
* mode:c
|
||||||
|
* indent-tabs-mode:t
|
||||||
|
* tab-width:4
|
||||||
|
* c-basic-offset:4
|
||||||
|
* End:
|
||||||
|
* For VIM:
|
||||||
|
* vim:set softtabstop=4 shiftwidth=4 tabstop=4 expandtab:
|
||||||
|
*/
|
60
src/mod/event_handlers/mod_cdr_csv/mod_cdr_csv.vcproj
Normal file
60
src/mod/event_handlers/mod_cdr_csv/mod_cdr_csv.vcproj
Normal file
@ -0,0 +1,60 @@
|
|||||||
|
<?xml version="1.0" encoding="Windows-1252"?>
|
||||||
|
<VisualStudioProject
|
||||||
|
ProjectType="Visual C++"
|
||||||
|
Version="8.00"
|
||||||
|
Name="mod_cdr_csv"
|
||||||
|
ProjectGUID="{08DAD348-9E0A-4A2E-97F1-F1E7E24A7836}"
|
||||||
|
RootNamespace="mod_cdr_csv"
|
||||||
|
Keyword="Win32Proj"
|
||||||
|
>
|
||||||
|
<Platforms>
|
||||||
|
<Platform
|
||||||
|
Name="Win32"
|
||||||
|
/>
|
||||||
|
</Platforms>
|
||||||
|
<ToolFiles>
|
||||||
|
</ToolFiles>
|
||||||
|
<Configurations>
|
||||||
|
<Configuration
|
||||||
|
Name="Debug|Win32"
|
||||||
|
ConfigurationType="2"
|
||||||
|
InheritedPropertySheets="..\..\..\..\w32\module_debug.vsprops;..\..\..\..\w32\curl.vsprops"
|
||||||
|
CharacterSet="2"
|
||||||
|
>
|
||||||
|
<Tool
|
||||||
|
Name="VCCLCompilerTool"
|
||||||
|
AdditionalIncludeDirectories=""
|
||||||
|
UsePrecompiledHeader="0"
|
||||||
|
/>
|
||||||
|
<Tool
|
||||||
|
Name="VCLinkerTool"
|
||||||
|
/>
|
||||||
|
</Configuration>
|
||||||
|
<Configuration
|
||||||
|
Name="Release|Win32"
|
||||||
|
ConfigurationType="2"
|
||||||
|
InheritedPropertySheets="..\..\..\..\w32\module_release.vsprops;..\..\..\..\w32\curl.vsprops"
|
||||||
|
CharacterSet="2"
|
||||||
|
>
|
||||||
|
<Tool
|
||||||
|
Name="VCCLCompilerTool"
|
||||||
|
AdditionalIncludeDirectories=""
|
||||||
|
UsePrecompiledHeader="0"
|
||||||
|
/>
|
||||||
|
<Tool
|
||||||
|
Name="VCLinkerTool"
|
||||||
|
AdditionalLibraryDirectories=""
|
||||||
|
/>
|
||||||
|
</Configuration>
|
||||||
|
</Configurations>
|
||||||
|
<References>
|
||||||
|
</References>
|
||||||
|
<Files>
|
||||||
|
<File
|
||||||
|
RelativePath=".\mod_cdr_csv.c"
|
||||||
|
>
|
||||||
|
</File>
|
||||||
|
</Files>
|
||||||
|
<Globals>
|
||||||
|
</Globals>
|
||||||
|
</VisualStudioProject>
|
@ -356,7 +356,7 @@ SWITCH_MODULE_LOAD_FUNCTION(mod_logfile_load)
|
|||||||
for (param = switch_xml_child(settings, "param"); param; param = param->next) {
|
for (param = switch_xml_child(settings, "param"); param; param = param->next) {
|
||||||
char *var = (char *) switch_xml_attr_soft(param, "name");
|
char *var = (char *) switch_xml_attr_soft(param, "name");
|
||||||
char *val = (char *) switch_xml_attr_soft(param, "value");
|
char *val = (char *) switch_xml_attr_soft(param, "value");
|
||||||
if (!strcmp(var, "rotate")) {
|
if (!strcmp(var, "rotate-on-hup")) {
|
||||||
globals.rotate = switch_true(val);
|
globals.rotate = switch_true(val);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -21,7 +21,7 @@
|
|||||||
* Portions created by the Initial Developer are Copyright (C)
|
* Portions created by the Initial Developer are Copyright (C)
|
||||||
* the Initial Developer. All Rights Reserved.
|
* the Initial Developer. All Rights Reserved.
|
||||||
*
|
*
|
||||||
* Contributor(s):
|
|
||||||
*
|
*
|
||||||
* Anthony Minessale II <anthmct@yahoo.com>
|
* Anthony Minessale II <anthmct@yahoo.com>
|
||||||
*
|
*
|
||||||
|
@ -327,13 +327,21 @@ SWITCH_DECLARE(const char *) switch_channel_get_variable(switch_channel_t *chann
|
|||||||
|
|
||||||
switch_mutex_lock(channel->profile_mutex);
|
switch_mutex_lock(channel->profile_mutex);
|
||||||
if (!(v = switch_event_get_header(channel->variables, (char*)varname))) {
|
if (!(v = switch_event_get_header(channel->variables, (char*)varname))) {
|
||||||
if (!channel->caller_profile || !(v = switch_caller_get_field_by_name(channel->caller_profile, varname))) {
|
switch_caller_profile_t *cp = channel->caller_profile;
|
||||||
if (!strcmp(varname, "base_dir")) {
|
|
||||||
v = SWITCH_GLOBAL_dirs.base_dir;
|
if (cp) {
|
||||||
} else {
|
if (!strncmp(varname, "aleg_", 5)) {
|
||||||
v = switch_core_get_variable(varname);
|
cp = cp->originator_caller_profile;
|
||||||
|
varname += 5;
|
||||||
|
} else if (!strncmp(varname, "bleg_", 5)) {
|
||||||
|
cp = cp->originatee_caller_profile;
|
||||||
|
varname += 5;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (!cp || !(v = switch_caller_get_field_by_name(cp, varname))) {
|
||||||
|
v = switch_core_get_variable(varname);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
switch_mutex_unlock(channel->profile_mutex);
|
switch_mutex_unlock(channel->profile_mutex);
|
||||||
|
|
||||||
@ -389,6 +397,7 @@ SWITCH_DECLARE(switch_status_t) switch_channel_set_name(switch_channel_t *channe
|
|||||||
if (name) {
|
if (name) {
|
||||||
char *uuid = switch_core_session_get_uuid(channel->session);
|
char *uuid = switch_core_session_get_uuid(channel->session);
|
||||||
channel->name = switch_core_session_strdup(channel->session, name);
|
channel->name = switch_core_session_strdup(channel->session, name);
|
||||||
|
switch_channel_set_variable(channel, SWITCH_CHANNEL_NAME_VARIABLE, name);
|
||||||
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_NOTICE, "New Chan %s [%s]\n", name, uuid);
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_NOTICE, "New Chan %s [%s]\n", name, uuid);
|
||||||
}
|
}
|
||||||
return SWITCH_STATUS_SUCCESS;
|
return SWITCH_STATUS_SUCCESS;
|
||||||
|
@ -678,6 +678,7 @@ SWITCH_DECLARE(switch_status_t) switch_core_init(switch_core_flag_t flags, switc
|
|||||||
switch_core_set_variable("local_ip_v4", guess_ip);
|
switch_core_set_variable("local_ip_v4", guess_ip);
|
||||||
switch_find_local_ip(guess_ip, sizeof(guess_ip), AF_INET6);
|
switch_find_local_ip(guess_ip, sizeof(guess_ip), AF_INET6);
|
||||||
switch_core_set_variable("local_ip_v6", guess_ip);
|
switch_core_set_variable("local_ip_v6", guess_ip);
|
||||||
|
switch_core_set_variable("base_dir", SWITCH_GLOBAL_dirs.base_dir);
|
||||||
|
|
||||||
|
|
||||||
if (switch_xml_init(runtime.memory_pool, err) != SWITCH_STATUS_SUCCESS) {
|
if (switch_xml_init(runtime.memory_pool, err) != SWITCH_STATUS_SUCCESS) {
|
||||||
@ -810,6 +811,20 @@ static void handle_SIGINT(int sig)
|
|||||||
if (sig);
|
if (sig);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void handle_SIGHUP(int sig)
|
||||||
|
{
|
||||||
|
if (sig) {
|
||||||
|
switch_event_t *event;
|
||||||
|
|
||||||
|
if (switch_event_create(&event, SWITCH_EVENT_TRAP) == SWITCH_STATUS_SUCCESS) {
|
||||||
|
switch_event_add_header(event, SWITCH_STACK_BOTTOM, "Trapped-Signal", "HUP");
|
||||||
|
switch_event_fire(&event);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
SWITCH_DECLARE(switch_status_t) switch_core_init_and_modload(switch_core_flag_t flags, switch_bool_t console, const char **err)
|
SWITCH_DECLARE(switch_status_t) switch_core_init_and_modload(switch_core_flag_t flags, switch_bool_t console, const char **err)
|
||||||
{
|
{
|
||||||
switch_event_t *event;
|
switch_event_t *event;
|
||||||
@ -832,6 +847,8 @@ SWITCH_DECLARE(switch_status_t) switch_core_init_and_modload(switch_core_flag_t
|
|||||||
signal(SIGBUS, handle_SIGBUS);
|
signal(SIGBUS, handle_SIGBUS);
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
signal(SIGHUP, handle_SIGHUP);
|
||||||
|
|
||||||
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_CONSOLE, "Bringing up environment.\n");
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_CONSOLE, "Bringing up environment.\n");
|
||||||
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_CONSOLE, "Loading Modules.\n");
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_CONSOLE, "Loading Modules.\n");
|
||||||
if (switch_loadable_module_init() != SWITCH_STATUS_SUCCESS) {
|
if (switch_loadable_module_init() != SWITCH_STATUS_SUCCESS) {
|
||||||
|
Loading…
x
Reference in New Issue
Block a user