diff --git a/src/mod/event_handlers/mod_json_cdr/Makefile b/src/mod/event_handlers/mod_json_cdr/Makefile
new file mode 100644
index 0000000000..d5043e017d
--- /dev/null
+++ b/src/mod/event_handlers/mod_json_cdr/Makefile
@@ -0,0 +1,27 @@
+json-c=json-c-0.9
+BASE=../../../..
+WANT_CURL=yes
+
+JSON_DIR=$(switch_srcdir)/libs/$(json-c)
+JSON_BUILDDIR=$(switch_builddir)/libs/$(json-c)
+
+JSONLA=$(JSON_BUILDDIR)/libjson.la
+
+LOCAL_CFLAGS=-I$(JSON_DIR)
+LOCAL_LIBADD=$(JSONLA)
+
+include $(BASE)/build/modmake.rules
+
+$(JSON_DIR):
+ $(GETLIB) $(json-c).tar.gz
+
+$(JSON_BUILDDIR)/Makefile: $(JSON_DIR)
+ mkdir -p $(JSON_BUILDDIR)
+ cd $(JSON_BUILDDIR) && $(DEFAULT_VARS) $(JSON_DIR)/configure $(DEFAULT_ARGS) --srcdir=$(JSON_DIR) CPPFLAGS= LDFLAGS=
+ $(TOUCH_TARGET)
+
+$(JSONLA): $(JSON_BUILDDIR)/Makefile
+ cd $(JSON_BUILDDIR) && $(MAKE)
+ $(TOUCH_TARGET)
+
+
diff --git a/src/mod/event_handlers/mod_json_cdr/mod_json_cdr.2008.vcproj b/src/mod/event_handlers/mod_json_cdr/mod_json_cdr.2008.vcproj
new file mode 100644
index 0000000000..b126d44ffb
--- /dev/null
+++ b/src/mod/event_handlers/mod_json_cdr/mod_json_cdr.2008.vcproj
@@ -0,0 +1,289 @@
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
diff --git a/src/mod/event_handlers/mod_json_cdr/mod_json_cdr.c b/src/mod/event_handlers/mod_json_cdr/mod_json_cdr.c
new file mode 100644
index 0000000000..b4f3eeceff
--- /dev/null
+++ b/src/mod/event_handlers/mod_json_cdr/mod_json_cdr.c
@@ -0,0 +1,1046 @@
+/*
+ * FreeSWITCH Modular Media Switching Software Library / Soft-Switch Application
+ * Copyright (C) 2005-2010, Anthony Minessale II
+ *
+ * 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
+ * Portions created by the Initial Developer are Copyright (C)
+ * the Initial Developer. All Rights Reserved.
+ *
+ * Contributor(s):
+ *
+ * Brian West
+ * Bret McDanel
+ * Justin Cassidy
+ *
+ * mod_json_cdr.c -- JSON CDR Module to files or curl
+ *
+ */
+#include
+#include
+#include
+#include
+
+#define MAX_URLS 20
+
+#define ENCODING_NONE 0
+#define ENCODING_DEFAULT 1
+#define ENCODING_BASE64 2
+
+static struct {
+ char *cred;
+ char *urls[MAX_URLS + 1];
+ int url_count;
+ int url_index;
+ switch_thread_rwlock_t *log_path_lock;
+ char *base_log_dir;
+ char *base_err_log_dir;
+ char *log_dir;
+ char *err_log_dir;
+ uint32_t delay;
+ uint32_t retries;
+ uint32_t shutdown;
+ uint32_t enable_cacert_check;
+ char *ssl_cert_file;
+ char *ssl_key_file;
+ char *ssl_key_password;
+ char *ssl_version;
+ char *ssl_cacert_file;
+ uint32_t enable_ssl_verifyhost;
+ int encode;
+ int log_http_and_disk;
+ int log_b;
+ int prefix_a;
+ int disable100continue;
+ int rotate;
+ int auth_scheme;
+ switch_memory_pool_t *pool;
+ switch_event_node_t *node;
+} globals;
+
+SWITCH_MODULE_LOAD_FUNCTION(mod_json_cdr_load);
+SWITCH_MODULE_SHUTDOWN_FUNCTION(mod_json_cdr_shutdown);
+SWITCH_MODULE_DEFINITION(mod_json_cdr, mod_json_cdr_load, mod_json_cdr_shutdown, NULL);
+
+/* this function would have access to the HTML returned by the webserver, we don't need it
+ * and the default curl activity is to print to stdout, something not as desirable
+ * so we have a dummy function here
+ */
+static size_t httpCallBack(char *buffer, size_t size, size_t nitems, void *outstream)
+{
+ return size * nitems;
+}
+
+static switch_status_t set_json_cdr_log_dirs()
+{
+ switch_time_exp_t tm;
+ char *path = NULL;
+ char date[80] = "";
+ switch_size_t retsize;
+ switch_status_t status = SWITCH_STATUS_SUCCESS, dir_status;
+
+ switch_time_exp_lt(&tm, switch_micro_time_now());
+ switch_strftime_nocheck(date, &retsize, sizeof(date), "%Y-%m-%d-%H-%M-%S", &tm);
+
+ switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_NOTICE, "Rotating log file paths\n");
+
+ if (!zstr(globals.base_log_dir)) {
+ if (globals.rotate) {
+ if ((path = switch_mprintf("%s%s%s", globals.base_log_dir, SWITCH_PATH_SEPARATOR, date))) {
+ switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_NOTICE, "Rotating log file path to %s\n", path);
+
+ dir_status = SWITCH_STATUS_SUCCESS;
+ if (switch_directory_exists(path, globals.pool) != SWITCH_STATUS_SUCCESS) {
+ dir_status = switch_dir_make(path, SWITCH_FPROT_OS_DEFAULT, globals.pool);
+ }
+
+ if (dir_status == SWITCH_STATUS_SUCCESS) {
+ switch_thread_rwlock_wrlock(globals.log_path_lock);
+ switch_safe_free(globals.log_dir);
+ globals.log_dir = path;
+ switch_thread_rwlock_unlock(globals.log_path_lock);
+ } else {
+ switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_ERROR, "Failed to create new mod_json_cdr log_dir path\n");
+ switch_safe_free(path);
+ status = SWITCH_STATUS_FALSE;
+ }
+ } else {
+ switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_ERROR, "Failed to generate new mod_json_cdr log_dir path\n");
+ status = SWITCH_STATUS_FALSE;
+ }
+ } else {
+ switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_NOTICE, "Setting log file path to %s\n", globals.base_log_dir);
+ if ((path = switch_safe_strdup(globals.base_log_dir))) {
+ switch_thread_rwlock_wrlock(globals.log_path_lock);
+ switch_safe_free(globals.log_dir);
+ globals.log_dir = path;
+ switch_thread_rwlock_unlock(globals.log_path_lock);
+ } else {
+ switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_ERROR, "Failed to set log_dir path\n");
+ status = SWITCH_STATUS_FALSE;
+ }
+ }
+ }
+
+ if (!zstr(globals.base_err_log_dir)) {
+ if (globals.rotate) {
+ if ((path = switch_mprintf("%s%s%s", globals.base_err_log_dir, SWITCH_PATH_SEPARATOR, date))) {
+ switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_NOTICE, "Rotating err log file path to %s\n", path);
+
+ dir_status = SWITCH_STATUS_SUCCESS;
+ if (switch_directory_exists(path, globals.pool) != SWITCH_STATUS_SUCCESS) {
+ dir_status = switch_dir_make(path, SWITCH_FPROT_OS_DEFAULT, globals.pool);
+ }
+
+ if (dir_status == SWITCH_STATUS_SUCCESS) {
+ switch_thread_rwlock_wrlock(globals.log_path_lock);
+ switch_safe_free(globals.err_log_dir);
+ globals.err_log_dir = path;
+ switch_thread_rwlock_unlock(globals.log_path_lock);
+ } else {
+ switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_ERROR, "Failed to create new mod_json_cdr err_log_dir path\n");
+ switch_safe_free(path);
+ status = SWITCH_STATUS_FALSE;
+ }
+ } else {
+ switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_ERROR, "Failed to generate new mod_json_cdr err_log_dir path\n");
+ status = SWITCH_STATUS_FALSE;
+ }
+ } else {
+ switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_NOTICE, "Setting err log file path to %s\n", globals.base_err_log_dir);
+ if ((path = switch_safe_strdup(globals.base_err_log_dir))) {
+ switch_thread_rwlock_wrlock(globals.log_path_lock);
+ switch_safe_free(globals.err_log_dir);
+ globals.err_log_dir = path;
+ switch_thread_rwlock_unlock(globals.log_path_lock);
+ } else {
+ switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_ERROR, "Failed to set err_log_dir path\n");
+ status = SWITCH_STATUS_FALSE;
+ }
+ }
+ }
+
+ return status;
+}
+
+#define JSON_ENSURE_SUCCESS(obj) if (is_error(obj)) { return; }
+SWITCH_DECLARE(void) set_json_profile_data(struct json_object *json, switch_caller_profile_t *caller_profile)
+{
+ struct json_object *param = NULL;
+
+ param = json_object_new_string((char *)caller_profile->username);
+ JSON_ENSURE_SUCCESS(param);
+ json_object_object_add(json, "username", param);
+
+ param = json_object_new_string((char *)caller_profile->dialplan);
+ JSON_ENSURE_SUCCESS(param);
+ json_object_object_add(json, "dialplan", param);
+
+ param = json_object_new_string((char *)caller_profile->caller_id_name);
+ JSON_ENSURE_SUCCESS(param);
+ json_object_object_add(json, "caller_id_name", param);
+
+ param = json_object_new_string((char *)caller_profile->ani);
+ JSON_ENSURE_SUCCESS(param);
+ json_object_object_add(json, "ani", param);
+
+ param = json_object_new_string((char *)caller_profile->aniii);
+ JSON_ENSURE_SUCCESS(param);
+ json_object_object_add(json, "aniii", param);
+
+ param = json_object_new_string((char *)caller_profile->caller_id_number);
+ JSON_ENSURE_SUCCESS(param);
+ json_object_object_add(json, "caller_id_number", param);
+
+ param = json_object_new_string((char *)caller_profile->network_addr);
+ JSON_ENSURE_SUCCESS(param);
+ json_object_object_add(json, "network_addr", param);
+
+ param = json_object_new_string((char *)caller_profile->rdnis);
+ JSON_ENSURE_SUCCESS(param);
+ json_object_object_add(json, "rdnis", param);
+
+ param = json_object_new_string(caller_profile->destination_number);
+ JSON_ENSURE_SUCCESS(param);
+ json_object_object_add(json, "destination_number", param);
+
+ param = json_object_new_string(caller_profile->uuid);
+ JSON_ENSURE_SUCCESS(param);
+ json_object_object_add(json, "uuid", param);
+
+ param = json_object_new_string((char *)caller_profile->source);
+ JSON_ENSURE_SUCCESS(param);
+ json_object_object_add(json, "source", param);
+
+ param = json_object_new_string((char *)caller_profile->context);
+ JSON_ENSURE_SUCCESS(param);
+ json_object_object_add(json, "context", param);
+
+ param = json_object_new_string(caller_profile->chan_name);
+ JSON_ENSURE_SUCCESS(param);
+ json_object_object_add(json, "chan_name", param);
+
+}
+
+SWITCH_DECLARE(void) set_json_chan_vars(struct json_object *json, switch_channel_t *channel)
+{
+ struct json_object *variable = NULL;
+ switch_event_header_t *hi = switch_channel_variable_first(channel);
+
+ if (!hi)
+ return;
+
+ for (; hi; hi = hi->next) {
+ if (!zstr(hi->name) && !zstr(hi->value)) {
+ char *data;
+ switch_size_t dlen = strlen(hi->value) * 3;
+
+ if ((data = malloc(dlen))) {
+ memset(data, 0, dlen);
+ switch_url_encode(hi->value, data, dlen);
+
+ variable = json_object_new_string(data);
+ if (!is_error(variable)) {
+ json_object_object_add(json, hi->name, variable);
+ }
+ free(data);
+ }
+ }
+ }
+ switch_channel_variable_last(channel);
+
+ return;
+}
+
+
+
+SWITCH_DECLARE(switch_status_t) generate_json_cdr(switch_core_session_t *session, struct json_object **json_cdr)
+{
+
+ struct json_object *cdr = json_object_new_object();
+ switch_channel_t *channel = switch_core_session_get_channel(session);
+ switch_caller_profile_t *caller_profile;
+ struct json_object *variables, *j_main_cp, *j_caller_profile, *j_caller_extension, *j_times, *time_tag,
+ *j_application, *j_callflow, *j_inner_extension, *j_apps, *j_o, *j_channel_data, *j_field;
+ switch_app_log_t *app_log;
+ char tmp[512], *f;
+
+ if (is_error(cdr)) {
+ return SWITCH_STATUS_FALSE;
+ }
+
+ j_channel_data = json_object_new_object();
+ if (is_error(j_channel_data)) {
+ goto error;
+ }
+ json_object_object_add(cdr, "channel_data", j_channel_data);
+
+
+ j_field = json_object_new_string((char *) switch_channel_state_name(switch_channel_get_state(channel)));
+ json_object_object_add(j_channel_data, "state", j_field);
+
+ j_field = json_object_new_string(switch_channel_direction(channel) == SWITCH_CALL_DIRECTION_OUTBOUND ? "outbound" : "inbound");
+
+ json_object_object_add(j_channel_data, "direction", j_field);
+
+
+ switch_snprintf(tmp, sizeof(tmp), "%d", switch_channel_get_state(channel));
+ j_field = json_object_new_string((char *) tmp);
+ json_object_object_add(j_channel_data, "state_number", j_field);
+
+
+ if ((f = switch_channel_get_flag_string(channel))) {
+ j_field = json_object_new_string((char *) f);
+ json_object_object_add(j_channel_data, "flags", j_field);
+ free(f);
+ }
+
+ if ((f = switch_channel_get_cap_string(channel))) {
+ j_field = json_object_new_string((char *) f);
+ json_object_object_add(j_channel_data, "caps", j_field);
+ free(f);
+ }
+
+
+ variables = json_object_new_object();
+ json_object_object_add(cdr, "variables", variables);
+
+ if (is_error(variables)) {
+ goto error;
+ }
+
+ set_json_chan_vars(variables, channel);
+
+
+ if ((app_log = switch_core_session_get_app_log(session))) {
+ switch_app_log_t *ap;
+
+ j_apps = json_object_new_object();
+
+ if (is_error(j_apps)) {
+ goto error;
+ }
+
+ json_object_object_add(cdr, "app_log", j_apps);
+
+ for (ap = app_log; ap; ap = ap->next) {
+ j_application = json_object_new_object();
+
+ if (is_error(j_application)) {
+ goto error;
+ }
+
+ json_object_object_add(j_application, "app_name", json_object_new_string(ap->app));
+ json_object_object_add(j_application, "app_data", json_object_new_string(ap->arg));
+
+ json_object_object_add(j_apps, "application", j_application);
+ }
+ }
+
+
+ caller_profile = switch_channel_get_caller_profile(channel);
+
+ while (caller_profile) {
+
+ j_callflow = json_object_new_object();
+
+ if (is_error(j_callflow)) {
+ goto error;
+ }
+
+ json_object_object_add(cdr, "callflow", j_callflow);
+
+ if (!zstr(caller_profile->dialplan)) {
+ json_object_object_add(j_callflow, "dialplan", json_object_new_string((char *)caller_profile->dialplan));
+ }
+
+ if (!zstr(caller_profile->profile_index)) {
+ json_object_object_add(j_callflow, "profile_index", json_object_new_string((char *)caller_profile->profile_index));
+ }
+
+ if (caller_profile->caller_extension) {
+ switch_caller_application_t *ap;
+
+ j_caller_extension = json_object_new_object();
+
+ if (is_error(j_caller_extension)) {
+ goto error;
+ }
+
+ json_object_object_add(j_callflow, "extension", j_caller_extension);
+
+ json_object_object_add(j_caller_extension, "name", json_object_new_string(caller_profile->caller_extension->extension_name));
+ json_object_object_add(j_caller_extension, "number", json_object_new_string(caller_profile->caller_extension->extension_number));
+
+ if (caller_profile->caller_extension->current_application) {
+ json_object_object_add(j_caller_extension, "current_app", json_object_new_string(caller_profile->caller_extension->current_application->application_name));
+ }
+
+ for (ap = caller_profile->caller_extension->applications; ap; ap = ap->next) {
+ j_application = json_object_new_object();
+
+ if (is_error(j_application)) {
+ goto error;
+ }
+
+
+ json_object_object_add(j_caller_extension, "application", j_application);
+
+ if (ap == caller_profile->caller_extension->current_application) {
+ json_object_object_add(j_application, "last_executed", json_object_new_string("true"));
+ }
+ json_object_object_add(j_application, "app_name", json_object_new_string(ap->application_name));
+ json_object_object_add(j_application, "app_data", json_object_new_string(ap->application_data));
+ }
+
+ if (caller_profile->caller_extension->children) {
+ switch_caller_profile_t *cp = NULL;
+ for (cp = caller_profile->caller_extension->children; cp; cp = cp->next) {
+
+ if (!cp->caller_extension) {
+ continue;
+ }
+
+ j_inner_extension = json_object_new_object();
+ if (is_error(j_inner_extension)) {
+ goto error;
+ }
+
+ json_object_object_add(j_caller_extension, "sub_extensions", j_inner_extension);
+
+
+ j_caller_extension = json_object_new_object();
+ if (is_error(j_caller_extension)) {
+ goto error;
+ }
+
+ json_object_object_add(j_inner_extension, "extension", j_caller_extension);
+
+ json_object_object_add(j_caller_extension, "name", json_object_new_string(cp->caller_extension->extension_name));
+ json_object_object_add(j_caller_extension, "number", json_object_new_string(cp->caller_extension->extension_number));
+
+ json_object_object_add(j_caller_extension, "dialplan", json_object_new_string((char *)cp->dialplan));
+
+ if (cp->caller_extension->current_application) {
+ json_object_object_add(j_caller_extension, "current_app", json_object_new_string(cp->caller_extension->current_application->application_name));
+ }
+
+ for (ap = cp->caller_extension->applications; ap; ap = ap->next) {
+ j_application = json_object_new_object();
+
+ if (is_error(j_application)) {
+ goto error;
+ }
+ json_object_object_add(j_caller_extension, "application", j_application);
+
+ if (ap == cp->caller_extension->current_application) {
+ json_object_object_add(j_application, "last_executed", json_object_new_string("true"));
+ }
+ json_object_object_add(j_application, "app_name", json_object_new_string(ap->application_name));
+ json_object_object_add(j_application, "app_data", json_object_new_string(ap->application_data));
+ }
+ }
+ }
+ }
+
+ j_main_cp = json_object_new_object();
+ if (is_error(j_main_cp)) {
+ goto error;
+ }
+
+ json_object_object_add(j_callflow, "caller_profile", j_main_cp);
+
+ set_json_profile_data(j_main_cp, caller_profile);
+
+ if (caller_profile->originator_caller_profile) {
+ switch_caller_profile_t *cp = NULL;
+
+ j_o = json_object_new_object();
+ if (is_error(j_o)) {
+ goto error;
+ }
+
+ json_object_object_add(j_main_cp, "originator", j_o);
+
+ for (cp = caller_profile->originator_caller_profile; cp; cp = cp->next) {
+ j_caller_profile = json_object_new_object();
+ if (is_error(j_caller_profile)) {
+ goto error;
+ }
+
+ json_object_object_add(j_o, "originator_caller_profile", j_caller_profile);
+
+ set_json_profile_data(j_caller_profile, cp);
+ }
+ }
+
+ if (caller_profile->originatee_caller_profile) {
+ switch_caller_profile_t *cp = NULL;
+
+ j_o = json_object_new_object();
+ if (is_error(j_o)) {
+ goto error;
+ }
+
+ json_object_object_add(j_main_cp, "originatee", j_o);
+
+ for (cp = caller_profile->originatee_caller_profile; cp; cp = cp->next) {
+
+ j_caller_profile = json_object_new_object();
+ if (is_error(j_caller_profile)) {
+ goto error;
+ }
+
+ json_object_object_add(j_o, "originatee_caller_profile", j_caller_profile);
+ set_json_profile_data(j_caller_profile, cp);
+ }
+ }
+
+ if (caller_profile->times) {
+
+ j_times = json_object_new_object();
+ if (is_error(j_times)) {
+ goto error;
+ }
+
+ json_object_object_add(j_callflow, "times", j_times);
+
+ switch_snprintf(tmp, sizeof(tmp), "%" SWITCH_TIME_T_FMT, caller_profile->times->created);
+ time_tag = json_object_new_string(tmp);
+ if (is_error(time_tag)) {
+ goto error;
+ }
+ json_object_object_add(j_times, "created_time", time_tag);
+
+ switch_snprintf(tmp, sizeof(tmp), "%" SWITCH_TIME_T_FMT, caller_profile->times->profile_created);
+ time_tag = json_object_new_string(tmp);
+ if (is_error(time_tag)) {
+ goto error;
+ }
+ json_object_object_add(j_times, "profile_created_time", time_tag);
+
+ switch_snprintf(tmp, sizeof(tmp), "%" SWITCH_TIME_T_FMT, caller_profile->times->progress);
+ time_tag = json_object_new_string(tmp);
+ if (is_error(time_tag)) {
+ goto error;
+ }
+ json_object_object_add(j_times, "progress_time", time_tag);
+
+ switch_snprintf(tmp, sizeof(tmp), "%" SWITCH_TIME_T_FMT, caller_profile->times->progress_media);
+ time_tag = json_object_new_string(tmp);
+ if (is_error(time_tag)) {
+ goto error;
+ }
+ json_object_object_add(j_times, "progress_media_time", time_tag);
+
+
+ switch_snprintf(tmp, sizeof(tmp), "%" SWITCH_TIME_T_FMT, caller_profile->times->answered);
+ time_tag = json_object_new_string(tmp);
+ if (is_error(time_tag)) {
+ goto error;
+ }
+ json_object_object_add(j_times, "answered_time", time_tag);
+
+ switch_snprintf(tmp, sizeof(tmp), "%" SWITCH_TIME_T_FMT, caller_profile->times->hungup);
+ time_tag = json_object_new_string(tmp);
+ if (is_error(time_tag)) {
+ goto error;
+ }
+ json_object_object_add(j_times, "hangup_time", time_tag);
+
+ switch_snprintf(tmp, sizeof(tmp), "%" SWITCH_TIME_T_FMT, caller_profile->times->resurrected);
+ time_tag = json_object_new_string(tmp);
+ if (is_error(time_tag)) {
+ goto error;
+ }
+ json_object_object_add(j_times, "resurrect_time", time_tag);
+
+ switch_snprintf(tmp, sizeof(tmp), "%" SWITCH_TIME_T_FMT, caller_profile->times->transferred);
+ time_tag = json_object_new_string(tmp);
+ if (is_error(time_tag)) {
+ goto error;
+ }
+ json_object_object_add(j_times, "transfer_time", time_tag);
+
+ }
+
+ caller_profile = caller_profile->next;
+ }
+
+ *json_cdr = cdr;
+
+ return SWITCH_STATUS_SUCCESS;
+
+ error:
+
+ if (cdr) {
+ json_object_put(cdr);
+ }
+
+ return SWITCH_STATUS_FALSE;
+}
+
+
+static switch_status_t my_on_reporting(switch_core_session_t *session)
+{
+ struct json_object *json_cdr = NULL;
+ const char *json_text = NULL;
+ char *path = NULL;
+ char *curl_json_text = NULL;
+ const char *logdir = NULL;
+ char *json_text_escaped = NULL;
+ int fd = -1;
+ uint32_t cur_try;
+ long httpRes;
+ CURL *curl_handle = NULL;
+ struct curl_slist *headers = NULL;
+ struct curl_slist *slist = NULL;
+ switch_channel_t *channel = switch_core_session_get_channel(session);
+ switch_status_t status = SWITCH_STATUS_FALSE;
+ int is_b;
+ const char *a_prefix = "";
+
+ if (globals.shutdown) {
+ return SWITCH_STATUS_SUCCESS;
+ }
+
+ is_b = channel && switch_channel_get_originator_caller_profile(channel);
+ if (!globals.log_b && is_b) {
+ return SWITCH_STATUS_SUCCESS;
+ }
+ if (!is_b && globals.prefix_a)
+ a_prefix = "a_";
+
+
+ if (generate_json_cdr(session, &json_cdr) != SWITCH_STATUS_SUCCESS) {
+ switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_ERROR, "Error Generating Data!\n");
+ return SWITCH_STATUS_FALSE;
+ }
+
+ json_text = json_object_to_json_string(json_cdr);
+
+ if (!json_text) {
+ switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_CRIT, "Memory Error!\n");
+ goto error;
+ }
+
+ switch_thread_rwlock_rdlock(globals.log_path_lock);
+
+ if (!(logdir = switch_channel_get_variable(channel, "json_cdr_base"))) {
+ logdir = globals.log_dir;
+ }
+
+ if (!zstr(logdir) && (globals.log_http_and_disk || !globals.url_count)) {
+ path = switch_mprintf("%s%s%s%s.cdr.json", logdir, SWITCH_PATH_SEPARATOR, a_prefix, switch_core_session_get_uuid(session));
+ switch_thread_rwlock_unlock(globals.log_path_lock);
+ if (path) {
+#ifdef _MSC_VER
+ if ((fd = open(path, O_WRONLY | O_CREAT | O_TRUNC, S_IRUSR | S_IWUSR)) > -1) {
+#else
+ if ((fd = open(path, O_WRONLY | O_CREAT | O_TRUNC, S_IRUSR | S_IWUSR | S_IRGRP | S_IWGRP | S_IROTH | S_IWOTH)) > -1) {
+#endif
+ int wrote;
+ wrote = write(fd, json_text, (unsigned) strlen(json_text));
+ close(fd);
+ fd = -1;
+ } else {
+ char ebuf[512] = { 0 };
+#ifdef WIN32
+ strerror_s(ebuf, sizeof(ebuf), errno);
+#else
+ strerror_r(errno, ebuf, sizeof(ebuf));
+#endif
+ switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_ERROR, "Error writing [%s][%s]\n", path, ebuf);
+ }
+ switch_safe_free(path);
+ }
+ } else {
+ switch_thread_rwlock_unlock(globals.log_path_lock);
+ }
+
+ /* try to post it to the web server */
+ if (globals.url_count) {
+ char *destUrl = NULL;
+ curl_handle = curl_easy_init();
+
+ if (globals.encode) {
+ switch_size_t need_bytes = strlen(json_text) * 3;
+
+ json_text_escaped = malloc(need_bytes);
+ switch_assert(json_text_escaped);
+ memset(json_text_escaped, 0, need_bytes);
+ if (globals.encode == ENCODING_DEFAULT) {
+ headers = curl_slist_append(headers, "Content-Type: application/x-www-form-urlencoded");
+ switch_url_encode(json_text, json_text_escaped, need_bytes);
+ } else {
+ headers = curl_slist_append(headers, "Content-Type: application/x-www-form-base64-encoded");
+ switch_b64_encode((unsigned char *) json_text, need_bytes / 3, (unsigned char *) json_text_escaped, need_bytes);
+ }
+
+ json_text = json_text_escaped;
+
+ if (!(curl_json_text = switch_mprintf("cdr=%s", json_text))) {
+ switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_CRIT, "Memory Error!\n");
+ goto error;
+ }
+
+ } else {
+ headers = curl_slist_append(headers, "Content-Type: application/x-www-form-plaintext");
+ curl_json_text = (char *)json_text;
+ }
+
+
+ if (!zstr(globals.cred)) {
+ curl_easy_setopt(curl_handle, CURLOPT_HTTPAUTH, globals.auth_scheme);
+ curl_easy_setopt(curl_handle, CURLOPT_USERPWD, globals.cred);
+ }
+
+ curl_easy_setopt(curl_handle, CURLOPT_HTTPHEADER, headers);
+ curl_easy_setopt(curl_handle, CURLOPT_POST, 1);
+ curl_easy_setopt(curl_handle, CURLOPT_POSTFIELDS, curl_json_text);
+ curl_easy_setopt(curl_handle, CURLOPT_USERAGENT, "freeswitch-json/1.0");
+ curl_easy_setopt(curl_handle, CURLOPT_WRITEFUNCTION, httpCallBack);
+
+ if (globals.disable100continue) {
+ slist = curl_slist_append(slist, "Expect:");
+ curl_easy_setopt(curl_handle, CURLOPT_HTTPHEADER, slist);
+ }
+
+ if (globals.ssl_cert_file) {
+ curl_easy_setopt(curl_handle, CURLOPT_SSLCERT, globals.ssl_cert_file);
+ }
+
+ if (globals.ssl_key_file) {
+ curl_easy_setopt(curl_handle, CURLOPT_SSLKEY, globals.ssl_key_file);
+ }
+
+ if (globals.ssl_key_password) {
+ curl_easy_setopt(curl_handle, CURLOPT_SSLKEYPASSWD, globals.ssl_key_password);
+ }
+
+ if (globals.ssl_version) {
+ if (!strcasecmp(globals.ssl_version, "SSLv3")) {
+ curl_easy_setopt(curl_handle, CURLOPT_SSLVERSION, CURL_SSLVERSION_SSLv3);
+ } else if (!strcasecmp(globals.ssl_version, "TLSv1")) {
+ curl_easy_setopt(curl_handle, CURLOPT_SSLVERSION, CURL_SSLVERSION_TLSv1);
+ }
+ }
+
+ if (globals.ssl_cacert_file) {
+ curl_easy_setopt(curl_handle, CURLOPT_CAINFO, globals.ssl_cacert_file);
+ }
+
+ /* these were used for testing, optionally they may be enabled if someone desires
+ curl_easy_setopt(curl_handle, CURLOPT_TIMEOUT, 120); // tcp timeout
+ curl_easy_setopt(curl_handle, CURLOPT_FOLLOWLOCATION, 1); // 302 recursion level
+ */
+
+ for (cur_try = 0; cur_try < globals.retries; cur_try++) {
+ if (cur_try > 0) {
+ switch_yield(globals.delay * 1000000);
+ }
+
+ destUrl = switch_mprintf("%s?uuid=%s", globals.urls[globals.url_index], switch_core_session_get_uuid(session));
+ curl_easy_setopt(curl_handle, CURLOPT_URL, destUrl);
+
+ if (!strncasecmp(destUrl, "https", 5)) {
+ curl_easy_setopt(curl_handle, CURLOPT_SSL_VERIFYPEER, 0);
+ curl_easy_setopt(curl_handle, CURLOPT_SSL_VERIFYHOST, 0);
+ }
+
+ if (globals.enable_cacert_check) {
+ curl_easy_setopt(curl_handle, CURLOPT_SSL_VERIFYPEER, TRUE);
+ }
+
+ if (globals.enable_ssl_verifyhost) {
+ curl_easy_setopt(curl_handle, CURLOPT_SSL_VERIFYHOST, 2);
+ }
+
+ curl_easy_perform(curl_handle);
+ curl_easy_getinfo(curl_handle, CURLINFO_RESPONSE_CODE, &httpRes);
+ switch_safe_free(destUrl);
+ if (httpRes == 200) {
+ goto success;
+ } else {
+ switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_ERROR, "Got error [%ld] posting to web server [%s]\n",
+ httpRes, globals.urls[globals.url_index]);
+ globals.url_index++;
+ switch_assert(globals.url_count <= MAX_URLS);
+ if (globals.url_index >= globals.url_count) {
+ globals.url_index = 0;
+ }
+ switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_ERROR, "Retry will be with url [%s]\n", globals.urls[globals.url_index]);
+ }
+ }
+ curl_easy_cleanup(curl_handle);
+ curl_slist_free_all(headers);
+ curl_slist_free_all(slist);
+ slist = NULL;
+ headers = NULL;
+ curl_handle = NULL;
+
+ /* if we are here the web post failed for some reason */
+ switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_ERROR, "Unable to post to web server, writing to file\n");
+
+ switch_thread_rwlock_rdlock(globals.log_path_lock);
+ path = switch_mprintf("%s%s%s%s.cdr.json", globals.err_log_dir, SWITCH_PATH_SEPARATOR, a_prefix, switch_core_session_get_uuid(session));
+ switch_thread_rwlock_unlock(globals.log_path_lock);
+ if (path) {
+#ifdef _MSC_VER
+ if ((fd = open(path, O_WRONLY | O_CREAT | O_TRUNC, S_IRUSR | S_IWUSR)) > -1) {
+#else
+ if ((fd = open(path, O_WRONLY | O_CREAT | O_TRUNC, S_IRUSR | S_IWUSR | S_IRGRP | S_IWGRP | S_IROTH | S_IWOTH)) > -1) {
+#endif
+ int wrote;
+ wrote = write(fd, json_text, (unsigned) strlen(json_text));
+ close(fd);
+ fd = -1;
+ } else {
+ char ebuf[512] = { 0 };
+#ifdef WIN32
+ strerror_s(ebuf, sizeof(ebuf), errno);
+#else
+ strerror_r(errno, ebuf, sizeof(ebuf));
+#endif
+ switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_ERROR, "Error![%s]\n", ebuf);
+ }
+ }
+ }
+
+ success:
+ status = SWITCH_STATUS_SUCCESS;
+
+ error:
+ if (curl_handle) {
+ curl_easy_cleanup(curl_handle);
+ }
+ if (headers) {
+ curl_slist_free_all(headers);
+ }
+ if (slist) {
+ curl_slist_free_all(slist);
+ }
+ if (curl_json_text != json_text) {
+ switch_safe_free(curl_json_text);
+ }
+
+ json_object_put(json_cdr);
+ switch_safe_free(json_text_escaped);
+
+ switch_safe_free(path);
+
+ return status;
+}
+
+static void event_handler(switch_event_t *event)
+{
+ const char *sig = switch_event_get_header(event, "Trapped-Signal");
+
+ if (sig && !strcmp(sig, "HUP")) {
+ if (globals.rotate) {
+ set_json_cdr_log_dirs();
+ }
+ }
+}
+
+static switch_state_handler_table_t state_handlers = {
+ /*.on_init */ NULL,
+ /*.on_routing */ NULL,
+ /*.on_execute */ NULL,
+ /*.on_hangup */ NULL,
+ /*.on_exchange_media */ NULL,
+ /*.on_soft_execute */ NULL,
+ /*.on_consume_media */ NULL,
+ /*.on_hibernate */ NULL,
+ /*.on_reset */ NULL,
+ /*.on_park */ NULL,
+ /*.on_reporting */ my_on_reporting
+};
+
+SWITCH_MODULE_LOAD_FUNCTION(mod_json_cdr_load)
+{
+ char *cf = "json_cdr.conf";
+ switch_xml_t cfg, xml, settings, param;
+ switch_status_t status = SWITCH_STATUS_SUCCESS;
+
+ /* test global state handlers */
+ switch_core_add_state_handler(&state_handlers);
+
+ *module_interface = switch_loadable_module_create_module_interface(pool, modname);
+
+ memset(&globals, 0, sizeof(globals));
+
+ if (switch_event_bind_removable(modname, SWITCH_EVENT_TRAP, SWITCH_EVENT_SUBCLASS_ANY, event_handler, NULL, &globals.node) != SWITCH_STATUS_SUCCESS) {
+ switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_ERROR, "Couldn't bind!\n");
+ return SWITCH_STATUS_GENERR;
+ }
+
+ globals.log_http_and_disk = 0;
+ globals.log_b = 1;
+ globals.disable100continue = 0;
+ globals.pool = pool;
+ globals.auth_scheme = CURLAUTH_BASIC;
+
+ switch_thread_rwlock_create(&globals.log_path_lock, pool);
+
+ /* parse the config */
+ if (!(xml = switch_xml_open_cfg(cf, &cfg, NULL))) {
+ switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_ERROR, "Open of %s failed\n", cf);
+ return SWITCH_STATUS_TERM;
+ }
+
+ 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, "cred") && !zstr(val)) {
+ globals.cred = switch_core_strdup(globals.pool, val);
+ } else if (!strcasecmp(var, "url") && !zstr(val)) {
+ if (globals.url_count >= MAX_URLS) {
+ switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_ERROR, "maximum urls configured!\n");
+ } else {
+ globals.urls[globals.url_count++] = switch_core_strdup(globals.pool, val);
+ }
+ } else if (!strcasecmp(var, "log-http-and-disk")) {
+ globals.log_http_and_disk = switch_true(val);
+ } else if (!strcasecmp(var, "delay") && !zstr(val)) {
+ globals.delay = (uint32_t) atoi(val);
+ } else if (!strcasecmp(var, "log-b-leg")) {
+ globals.log_b = switch_true(val);
+ } else if (!strcasecmp(var, "prefix-a-leg")) {
+ globals.prefix_a = switch_true(val);
+ } else if (!strcasecmp(var, "disable-100-continue") && switch_true(val)) {
+ globals.disable100continue = 1;
+ } else if (!strcasecmp(var, "encode") && !zstr(val)) {
+ if (!strcasecmp(val, "base64")) {
+ globals.encode = ENCODING_BASE64;
+ } else {
+ globals.encode = switch_true(val) ? ENCODING_DEFAULT : ENCODING_NONE;
+ }
+ } else if (!strcasecmp(var, "retries") && !zstr(val)) {
+ globals.retries = (uint32_t) atoi(val);
+ } else if (!strcasecmp(var, "rotate") && !zstr(val)) {
+ globals.rotate = switch_true(val);
+ } else if (!strcasecmp(var, "log-dir")) {
+ if (zstr(val)) {
+ globals.base_log_dir = switch_core_sprintf(globals.pool, "%s%sjson_cdr", SWITCH_GLOBAL_dirs.log_dir, SWITCH_PATH_SEPARATOR);
+ } else {
+ if (switch_is_file_path(val)) {
+ globals.base_log_dir = switch_core_strdup(globals.pool, val);
+ } else {
+ globals.base_log_dir = switch_core_sprintf(globals.pool, "%s%s%s", SWITCH_GLOBAL_dirs.log_dir, SWITCH_PATH_SEPARATOR, val);
+ }
+ }
+ } else if (!strcasecmp(var, "err-log-dir")) {
+ if (zstr(val)) {
+ globals.base_err_log_dir = switch_core_sprintf(globals.pool, "%s%sjson_cdr", SWITCH_GLOBAL_dirs.log_dir, SWITCH_PATH_SEPARATOR);
+ } else {
+ if (switch_is_file_path(val)) {
+ globals.base_err_log_dir = switch_core_strdup(globals.pool, val);
+ } else {
+ globals.base_err_log_dir = switch_core_sprintf(globals.pool, "%s%s%s", SWITCH_GLOBAL_dirs.log_dir, SWITCH_PATH_SEPARATOR, val);
+ }
+ }
+ } else if (!strcasecmp(var, "enable-cacert-check") && switch_true(val)) {
+ globals.enable_cacert_check = 1;
+ } else if (!strcasecmp(var, "ssl-cert-path")) {
+ globals.ssl_cert_file = val;
+ } else if (!strcasecmp(var, "ssl-key-path")) {
+ globals.ssl_key_file = val;
+ } else if (!strcasecmp(var, "ssl-key-password")) {
+ globals.ssl_key_password = val;
+ } else if (!strcasecmp(var, "ssl-version")) {
+ globals.ssl_version = val;
+ } else if (!strcasecmp(var, "ssl-cacert-file")) {
+ globals.ssl_cacert_file = val;
+ } else if (!strcasecmp(var, "enable-ssl-verifyhost") && switch_true(val)) {
+ globals.enable_ssl_verifyhost = 1;
+ } else if (!strcasecmp(var, "auth-scheme")) {
+ if (*val == '=') {
+ globals.auth_scheme = 0;
+ val++;
+ }
+
+ if (!strcasecmp(val, "basic")) {
+ globals.auth_scheme |= CURLAUTH_BASIC;
+ } else if (!strcasecmp(val, "digest")) {
+ globals.auth_scheme |= CURLAUTH_DIGEST;
+ } else if (!strcasecmp(val, "NTLM")) {
+ globals.auth_scheme |= CURLAUTH_NTLM;
+ } else if (!strcasecmp(val, "GSS-NEGOTIATE")) {
+ globals.auth_scheme |= CURLAUTH_GSSNEGOTIATE;
+ } else if (!strcasecmp(val, "any")) {
+ globals.auth_scheme = CURLAUTH_ANY;
+ }
+ }
+
+ }
+
+ if (zstr(globals.base_err_log_dir)) {
+ if (!zstr(globals.base_log_dir)) {
+ globals.base_err_log_dir = switch_core_strdup(globals.pool, globals.base_log_dir);
+ } else {
+ globals.base_err_log_dir = switch_core_sprintf(globals.pool, "%s%sjson_cdr", SWITCH_GLOBAL_dirs.log_dir, SWITCH_PATH_SEPARATOR);
+ }
+ }
+ }
+
+ if (globals.retries < 0) {
+ switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_ERROR, "Retries is negative, setting to 0\n");
+ globals.retries = 0;
+ }
+
+ if (globals.retries && globals.delay <= 0) {
+ switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_ERROR, "Retries set but delay 0 setting to 5000ms\n");
+ globals.delay = 5000;
+ }
+
+ globals.retries++;
+
+ set_json_cdr_log_dirs();
+
+ switch_xml_free(xml);
+ return status;
+}
+
+SWITCH_MODULE_SHUTDOWN_FUNCTION(mod_json_cdr_shutdown)
+{
+
+ globals.shutdown = 1;
+
+ switch_safe_free(globals.log_dir);
+ switch_safe_free(globals.err_log_dir);
+
+ switch_event_unbind(&globals.node);
+ switch_core_remove_state_handler(&state_handlers);
+
+ switch_thread_rwlock_destroy(globals.log_path_lock);
+
+ 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:
+ */
diff --git a/src/mod/event_handlers/mod_json_cdr/mod_json_cdr.vcproj b/src/mod/event_handlers/mod_json_cdr/mod_json_cdr.vcproj
new file mode 100644
index 0000000000..657493a9bb
--- /dev/null
+++ b/src/mod/event_handlers/mod_json_cdr/mod_json_cdr.vcproj
@@ -0,0 +1,60 @@
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+