/* * FreeSWITCH Modular Media Switching Software Library / Soft-Switch Application * Copyright (C) 2005/2006, 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): * * Anthony Minessale II * Michael Jerris * Johny Kadarisman * Paul Tinsley * Marcel Barbulescu * * * mod_commands.c -- Misc. Command Module * */ #include #include SWITCH_MODULE_LOAD_FUNCTION(mod_commands_load); SWITCH_MODULE_DEFINITION(mod_commands, mod_commands_load, NULL, NULL); static switch_api_interface_t ctl_api_interface; static switch_api_interface_t uuid_bridge_api_interface; static switch_api_interface_t session_record_api_interface; static switch_api_interface_t status_api_interface; static switch_api_interface_t show_api_interface; static switch_api_interface_t pause_api_interface; static switch_api_interface_t transfer_api_interface; static switch_api_interface_t load_api_interface; static switch_api_interface_t unload_api_interface; static switch_api_interface_t reload_api_interface; static switch_api_interface_t kill_api_interface; static switch_api_interface_t originate_api_interface; static switch_api_interface_t media_api_interface; static switch_api_interface_t hold_api_interface; static switch_api_interface_t broadcast_api_interface; static switch_api_interface_t sched_broadcast_api_interface; static switch_api_interface_t sched_transfer_api_interface; static switch_api_interface_t sched_hangup_api_interface; SWITCH_STANDARD_API(status_function) { uint8_t html = 0; switch_core_time_duration_t duration; char *http = NULL; if (session) { return SWITCH_STATUS_FALSE; } switch_core_measure_time(switch_core_uptime(), &duration); if (stream->event) { http = switch_event_get_header(stream->event, "http-host"); } if (http || (cmd && strstr(cmd, "html"))) { html = 1; stream->write_function(stream, "

FreeSWITCH Status

\n"); } stream->write_function(stream, "UP %u year%s, %u day%s, %u hour%s, %u minute%s, %u second%s, %u millisecond%s, %u microsecond%s\n", duration.yr, duration.yr == 1 ? "" : "s", duration.day, duration.day == 1 ? "" : "s", duration.hr, duration.hr == 1 ? "" : "s", duration.min, duration.min == 1 ? "" : "s", duration.sec, duration.sec == 1 ? "" : "s", duration.ms, duration.ms == 1 ? "" : "s", duration.mms, duration.mms == 1 ? "" : "s"); stream->write_function(stream, "%"SWITCH_SIZE_T_FMT" sessions since startup\n", switch_core_session_id() - 1 ); stream->write_function(stream, "%d sessions\n", switch_core_session_count()); if (html) { stream->write_function(stream, "\n"); } if (cmd && strstr(cmd, "refresh=")) { char *refresh = strchr(cmd, '='); if (refresh) { int r; refresh++; r = atoi(refresh); if (r > 0) { stream->write_function(stream, "\n", r, r, html ? "html=1" : ""); } } } return SWITCH_STATUS_SUCCESS; } SWITCH_STANDARD_API(ctl_function) { int argc; char *mydata, *argv[5]; uint32_t arg = 0; if (switch_strlen_zero(cmd)) { stream->write_function(stream, "USAGE: %s\n", ctl_api_interface.syntax); return SWITCH_STATUS_SUCCESS; } if ((mydata = strdup(cmd))) { argc = switch_separate_string(mydata, ' ', argv, (sizeof(argv) / sizeof(argv[0]))); if (!strcmp(argv[0], "hupall")) { arg = 1; switch_core_session_ctl(SCSC_HUPALL, &arg); } else if (!strcmp(argv[0], "pause")) { arg = 1; switch_core_session_ctl(SCSC_PAUSE_INBOUND, &arg); } else if (!strcmp(argv[0], "resume")) { arg = 0; switch_core_session_ctl(SCSC_PAUSE_INBOUND, &arg); } else if (!strcmp(argv[0], "shutdown")) { arg = 0; switch_core_session_ctl(SCSC_SHUTDOWN, &arg); } else { stream->write_function(stream, "INVALID COMMAND\nUSAGE: fsctl [hupall|pause|resume|shutdown]\n"); goto end; } stream->write_function(stream, "OK\n"); end: free(mydata); } else { stream->write_function(stream, "MEM ERR\n"); } return SWITCH_STATUS_SUCCESS; } SWITCH_STANDARD_API(load_function) { const char *err; if (session) { return SWITCH_STATUS_FALSE; } if (switch_strlen_zero(cmd)) { stream->write_function(stream, "USAGE: %s\n", load_api_interface.syntax); return SWITCH_STATUS_SUCCESS; } if (switch_loadable_module_load_module((char *) SWITCH_GLOBAL_dirs.mod_dir, (char *) cmd, SWITCH_TRUE, &err) == SWITCH_STATUS_SUCCESS) { stream->write_function(stream, "OK\n"); } else { stream->write_function(stream, "ERROR [%s]\n", err); } return SWITCH_STATUS_SUCCESS; } SWITCH_STANDARD_API(unload_function) { const char *err; if (session) { return SWITCH_STATUS_FALSE; } if (switch_strlen_zero(cmd)) { stream->write_function(stream, "USAGE: %s\n", unload_api_interface.syntax); return SWITCH_STATUS_SUCCESS; } if (switch_loadable_module_unload_module((char *) SWITCH_GLOBAL_dirs.mod_dir, (char *) cmd, &err) == SWITCH_STATUS_SUCCESS) { stream->write_function(stream, "OK\n"); } else { stream->write_function(stream, "ERROR [%s]\n", err); } return SWITCH_STATUS_SUCCESS; } SWITCH_STANDARD_API(reload_function) { const char *err; switch_xml_t xml_root; if (session) { return SWITCH_STATUS_FALSE; } if ((xml_root = switch_xml_open_root(1, &err))) { switch_xml_free(xml_root); } stream->write_function(stream, "OK [%s]\n", err); return SWITCH_STATUS_SUCCESS; } SWITCH_STANDARD_API(kill_function) { switch_core_session_t *ksession = NULL; if (session) { return SWITCH_STATUS_FALSE; } if (!cmd) { stream->write_function(stream, "USAGE: %s\n", kill_api_interface.syntax); } else if ((ksession = switch_core_session_locate(cmd))) { switch_channel_t *channel = switch_core_session_get_channel(ksession); switch_channel_hangup(channel, SWITCH_CAUSE_NORMAL_CLEARING); switch_core_session_rwunlock(ksession); stream->write_function(stream, "OK\n"); } else { stream->write_function(stream, "No Such Channel!\n"); } return SWITCH_STATUS_SUCCESS; } SWITCH_STANDARD_API(transfer_function) { switch_core_session_t *tsession = NULL; char *mycmd = NULL, *argv[4] = { 0 }; int argc = 0; if (session) { return SWITCH_STATUS_FALSE; } if (!switch_strlen_zero(cmd) && (mycmd = strdup(cmd))) { argc = switch_separate_string(mycmd, ' ', argv, (sizeof(argv) / sizeof(argv[0]))); if (argc >= 2 && argc <= 4) { char *uuid = argv[0]; char *dest = argv[1]; char *dp = argv[2]; char *context = argv[3]; if ((tsession = switch_core_session_locate(uuid))) { if (switch_ivr_session_transfer(tsession, dest, dp, context) == SWITCH_STATUS_SUCCESS) { stream->write_function(stream, "OK\n"); } else { stream->write_function(stream, "ERROR\n"); } switch_core_session_rwunlock(tsession); } else { stream->write_function(stream, "No Such Channel!\n"); } goto done; } } stream->write_function(stream, "USAGE: %s\n", transfer_api_interface.syntax); done: switch_safe_free(mycmd); return SWITCH_STATUS_SUCCESS; } SWITCH_STANDARD_API(sched_transfer_function) { switch_core_session_t *tsession = NULL; char *mycmd = NULL, *argv[6] = { 0 }; int argc = 0; if (session) { return SWITCH_STATUS_FALSE; } if (!switch_strlen_zero(cmd) && (mycmd = strdup(cmd))) { argc = switch_separate_string(mycmd, ' ', argv, (sizeof(argv) / sizeof(argv[0]))); } if (switch_strlen_zero(cmd) || argc < 2 || argc > 5) { stream->write_function(stream, "USAGE: %s\n", sched_transfer_api_interface.syntax); } else { char *uuid = argv[1]; char *dest = argv[2]; char *dp = argv[3]; char *context = argv[4]; time_t when; if (*argv[0] == '+') { when = time(NULL) + atol(argv[0] + 1); } else { when = atol(argv[0]); } if ((tsession = switch_core_session_locate(uuid))) { switch_ivr_schedule_transfer(when, uuid, dest, dp, context); stream->write_function(stream, "OK\n"); switch_core_session_rwunlock(tsession); } else { stream->write_function(stream, "No Such Channel!\n"); } } switch_safe_free(mycmd); return SWITCH_STATUS_SUCCESS; } SWITCH_STANDARD_API(sched_hangup_function) { switch_core_session_t *hsession = NULL; char *mycmd = NULL, *argv[4] = { 0 }; int argc = 0; if (session) { return SWITCH_STATUS_FALSE; } if (!switch_strlen_zero(cmd) && (mycmd = strdup(cmd))) { argc = switch_separate_string(mycmd, ' ', argv, (sizeof(argv) / sizeof(argv[0]))); } if (switch_strlen_zero(cmd) || argc < 1) { stream->write_function(stream, "USAGE: %s\n", sched_hangup_api_interface.syntax); } else { char *uuid = argv[1]; char *cause_str = argv[2]; time_t when; switch_call_cause_t cause = SWITCH_CAUSE_ALLOTTED_TIMEOUT; if (*argv[0] == '+') { when = time(NULL) + atol(argv[0] + 1); } else { when = atol(argv[0]); } if (cause_str) { cause = switch_channel_str2cause(cause_str); } if ((hsession = switch_core_session_locate(uuid))) { switch_ivr_schedule_hangup(when, uuid, cause, SWITCH_FALSE); stream->write_function(stream, "OK\n"); switch_core_session_rwunlock(hsession); } else { stream->write_function(stream, "No Such Channel!\n"); } } switch_safe_free(mycmd); return SWITCH_STATUS_SUCCESS; } SWITCH_STANDARD_API(uuid_media_function) { char *mycmd = NULL, *argv[4] = { 0 }; int argc = 0; switch_status_t status = SWITCH_STATUS_FALSE; if (session) { return status; } if (!switch_strlen_zero(cmd) && (mycmd = strdup(cmd))) { argc = switch_separate_string(mycmd, ' ', argv, (sizeof(argv) / sizeof(argv[0]))); } if (switch_strlen_zero(cmd) || argc < 1) { stream->write_function(stream, "USAGE: %s\n", media_api_interface.syntax); } else { if (!strcmp(argv[0], "off")) { status = switch_ivr_nomedia(argv[1], SMF_REBRIDGE); } else { status = switch_ivr_media(argv[0], SMF_REBRIDGE); } } if (status == SWITCH_STATUS_SUCCESS) { stream->write_function(stream, "+OK Success\n"); } else { stream->write_function(stream, "-ERR Operation Failed\n"); } switch_safe_free(mycmd); return SWITCH_STATUS_SUCCESS; } SWITCH_STANDARD_API(uuid_broadcast_function) { char *mycmd = NULL, *argv[4] = { 0 }; int argc = 0; switch_status_t status = SWITCH_STATUS_FALSE; if (session) { return status; } if (!switch_strlen_zero(cmd) && (mycmd = strdup(cmd))) { argc = switch_separate_string(mycmd, ' ', argv, (sizeof(argv) / sizeof(argv[0]))); } if (switch_strlen_zero(cmd) || argc < 2) { stream->write_function(stream, "USAGE: %s\n", broadcast_api_interface.syntax); } else { switch_media_flag_t flags = SMF_NONE; if (argv[2]) { if (!strcmp(argv[2], "both")) { flags |= (SMF_ECHO_ALEG | SMF_ECHO_BLEG); } else if (!strcmp(argv[2], "aleg")) { flags |= SMF_ECHO_ALEG; } else if (!strcmp(argv[2], "bleg")) { flags |= SMF_ECHO_BLEG; } } else { flags |= SMF_ECHO_ALEG; } status = switch_ivr_broadcast(argv[0], argv[1], flags); stream->write_function(stream, "+OK Message Sent\n"); } switch_safe_free(mycmd); return SWITCH_STATUS_SUCCESS; } SWITCH_STANDARD_API(sched_broadcast_function) { char *mycmd = NULL, *argv[4] = { 0 }; int argc = 0; switch_status_t status = SWITCH_STATUS_FALSE; if (session) { return status; } if (!switch_strlen_zero(cmd) && (mycmd = strdup(cmd))) { argc = switch_separate_string(mycmd, ' ', argv, (sizeof(argv) / sizeof(argv[0]))); } if (switch_strlen_zero(cmd) || argc < 3) { stream->write_function(stream, "USAGE: %s\n", sched_broadcast_api_interface.syntax); } else { switch_media_flag_t flags = SMF_NONE; time_t when; if (*argv[0] == '+') { when = time(NULL) + atol(argv[0] + 1); } else { when = atol(argv[0]); } if (argv[3]) { if (!strcmp(argv[3], "both")) { flags |= (SMF_ECHO_ALEG | SMF_ECHO_BLEG); } else if (!strcmp(argv[3], "aleg")) { flags |= SMF_ECHO_ALEG; } else if (!strcmp(argv[3], "bleg")) { flags |= SMF_ECHO_BLEG; } } else { flags |= SMF_ECHO_ALEG; } status = switch_ivr_schedule_broadcast(when, argv[1], argv[2], flags); stream->write_function(stream, "+OK Message Scheduled\n"); } switch_safe_free(mycmd); return SWITCH_STATUS_SUCCESS; } SWITCH_STANDARD_API(uuid_hold_function) { char *mycmd = NULL, *argv[4] = { 0 }; int argc = 0; switch_status_t status = SWITCH_STATUS_FALSE; if (session) { return status; } if (!switch_strlen_zero(cmd) && (mycmd = strdup(cmd))) { argc = switch_separate_string(mycmd, ' ', argv, (sizeof(argv) / sizeof(argv[0]))); } if (switch_strlen_zero(cmd) || argc < 1) { stream->write_function(stream, "USAGE: %s\n", hold_api_interface.syntax); } else { if (!strcmp(argv[0], "off")) { status = switch_ivr_unhold_uuid(argv[1]); } else { status = switch_ivr_hold_uuid(argv[0]); } } if (status == SWITCH_STATUS_SUCCESS) { stream->write_function(stream, "+OK Success\n"); } else { stream->write_function(stream, "-ERR Operation Failed\n"); } switch_safe_free(mycmd); return SWITCH_STATUS_SUCCESS; } SWITCH_STANDARD_API(uuid_bridge_function) { char *mycmd = NULL, *argv[4] = { 0 }; int argc = 0; if (session) { return SWITCH_STATUS_FALSE; } if (!switch_strlen_zero(cmd) && (mycmd = strdup(cmd))) { argc = switch_separate_string(mycmd, ' ', argv, (sizeof(argv) / sizeof(argv[0]))); } if (switch_strlen_zero(cmd) || argc != 2) { stream->write_function(stream, "USAGE: %s\n", uuid_bridge_api_interface.syntax); } else { if (switch_ivr_uuid_bridge(argv[0], argv[1]) != SWITCH_STATUS_SUCCESS) { stream->write_function(stream, "Invalid uuid\n"); } } switch_safe_free(mycmd); return SWITCH_STATUS_SUCCESS; } SWITCH_STANDARD_API(session_record_function) { switch_core_session_t *rsession = NULL; char *mycmd = NULL, *argv[4] = { 0 }; char *uuid = NULL, *action = NULL, *path = NULL; int argc = 0; uint32_t limit = 0; if (session) { return SWITCH_STATUS_FALSE; } if (switch_strlen_zero(cmd)) { goto usage; } if (!(mycmd = strdup(cmd))) { goto usage; } if ((argc = switch_separate_string(mycmd, ' ', argv, (sizeof(argv) / sizeof(argv[0])))) != 3) { goto usage; } uuid = argv[0]; action = argv[1]; path = argv[2]; limit = argv[3] ? atoi(argv[3]) : 0; if (!(rsession = switch_core_session_locate(uuid))) { stream->write_function(stream, "-Error Cannot locate session!\n"); return SWITCH_STATUS_SUCCESS; } if (switch_strlen_zero(action) || switch_strlen_zero(path)) { goto usage; } if (!strcasecmp(action, "start")) { switch_ivr_record_session(rsession, path, limit, NULL); } else if (!strcasecmp(action, "stop")) { switch_ivr_stop_record_session(rsession, path); } else { goto usage; } goto done; usage: stream->write_function(stream, "USAGE: %s\n", session_record_api_interface.syntax); switch_safe_free(mycmd); done: if (rsession) { switch_core_session_rwunlock(rsession); } switch_safe_free(mycmd); return SWITCH_STATUS_SUCCESS; } SWITCH_STANDARD_API(pause_function) { switch_core_session_t *psession = NULL; char *mycmd = NULL, *argv[4] = { 0 }; int argc = 0; if (session) { return SWITCH_STATUS_FALSE; } if (!switch_strlen_zero(cmd) && (mycmd = strdup(cmd))) { argc = switch_separate_string(mycmd, ' ', argv, (sizeof(argv) / sizeof(argv[0]))); } if (switch_strlen_zero(cmd) || argc < 2) { stream->write_function(stream, "USAGE: %s\n", pause_api_interface.syntax); } else { char *uuid = argv[0]; char *dest = argv[1]; if ((psession = switch_core_session_locate(uuid))) { switch_channel_t *channel = switch_core_session_get_channel(psession); if (!strcasecmp(dest, "on")) { switch_channel_set_flag(channel, CF_HOLD); } else { switch_channel_clear_flag(channel, CF_HOLD); } switch_core_session_rwunlock(psession); } else { stream->write_function(stream, "No Such Channel!\n"); } } switch_safe_free(mycmd); return SWITCH_STATUS_SUCCESS; } SWITCH_STANDARD_API(originate_function) { switch_channel_t *caller_channel; switch_core_session_t *caller_session = NULL; char *mycmd = NULL, *argv[7] = { 0 }; int i = 0, x, argc = 0; char *aleg, *exten, *dp, *context, *cid_name, *cid_num; uint32_t timeout = 60; switch_call_cause_t cause = SWITCH_CAUSE_NORMAL_CLEARING; uint8_t machine = 1; if (session) { stream->write_function(stream, "Illegal Usage\n"); return SWITCH_STATUS_SUCCESS; } if (!switch_strlen_zero(cmd) && (mycmd = strdup(cmd))) { argc = switch_separate_string(mycmd, ' ', argv, (sizeof(argv) / sizeof(argv[0]))); } if (switch_strlen_zero(cmd) || argc < 2 || argc > 7) { stream->write_function(stream, "USAGE: %s\n", originate_api_interface.syntax); switch_safe_free(mycmd); return SWITCH_STATUS_SUCCESS; } for (x = 0; x < argc; x++) { if (!strcasecmp(argv[x], "undef")) { argv[x] = NULL; } } if (!strcasecmp(argv[0], "machine")) { machine = 1; i++; } aleg = argv[i++]; exten = argv[i++]; dp = argv[i++]; context = argv[i++]; cid_name = argv[i++]; cid_num = argv[i++]; if (!dp) { dp = "XML"; } if (!context) { context = "default"; } if (argv[6]) { timeout = atoi(argv[6]); } if (switch_ivr_originate(NULL, &caller_session, &cause, aleg, timeout, NULL, cid_name, cid_num, NULL) != SWITCH_STATUS_SUCCESS) { if (machine) { stream->write_function(stream, "fail: %s\n", switch_channel_cause2str(cause)); } else { stream->write_function(stream, "Cannot Create Outgoing Channel! [%s] cause: %s\n", aleg, switch_channel_cause2str(cause)); } switch_safe_free(mycmd); return SWITCH_STATUS_SUCCESS; } caller_channel = switch_core_session_get_channel(caller_session); assert(caller_channel != NULL); switch_channel_clear_state_handler(caller_channel, NULL); if (*exten == '&' && *(exten + 1)) { switch_caller_extension_t *extension = NULL; char *app_name = switch_core_session_strdup(caller_session, (exten + 1)); char *arg = NULL, *e; if ((e = strchr(app_name, ')'))) { *e = '\0'; } if ((arg = strchr(app_name, '('))) { *arg++ = '\0'; } if ((extension = switch_caller_extension_new(caller_session, app_name, arg)) == 0) { switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_CRIT, "memory error!\n"); switch_channel_hangup(caller_channel, SWITCH_CAUSE_DESTINATION_OUT_OF_ORDER); switch_safe_free(mycmd); return SWITCH_STATUS_MEMERR; } switch_caller_extension_add_application(caller_session, extension, app_name, arg); switch_channel_set_caller_extension(caller_channel, extension); switch_channel_set_state(caller_channel, CS_EXECUTE); } else { switch_ivr_session_transfer(caller_session, exten, dp, context); } if (machine) { stream->write_function(stream, "success: %s\n", switch_core_session_get_uuid(caller_session)); } else { stream->write_function(stream, "Created Session: %s\n", switch_core_session_get_uuid(caller_session)); } if (caller_session) { switch_core_session_rwunlock(caller_session); } switch_safe_free(mycmd); return SWITCH_STATUS_SUCCESS; } static void sch_api_callback(switch_scheduler_task_t *task) { char *cmd, *arg = NULL; switch_stream_handle_t stream = { 0 }; assert(task); cmd = (char *) task->cmd_arg; if ((arg = strchr(cmd, ' '))) { *arg++ = '\0'; } SWITCH_STANDARD_STREAM(stream); switch_api_execute(cmd, arg, NULL, &stream); switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_DEBUG, "Command %s(%s):\n%s\n", cmd, arg, switch_str_nil((char *) stream.data)); switch_safe_free(stream.data); } SWITCH_STANDARD_API(sched_del_function) { uint32_t cnt = 0; if (switch_is_digit_string(cmd)) { int64_t tmp; tmp = (uint32_t) atoi(cmd); if (tmp > 0) { cnt = switch_scheduler_del_task_id((uint32_t)tmp); } } else { cnt = switch_scheduler_del_task_group(cmd); } stream->write_function(stream, "DELETED: %u\n", cnt); return SWITCH_STATUS_SUCCESS; } SWITCH_STANDARD_API(xml_wrap_api_function) { char *dcommand, *edata = NULL, *send = NULL, *command, *arg = NULL; switch_stream_handle_t mystream = { 0 }; int encoded = 0, elen = 0; if ((dcommand = strdup(cmd))) { if (!strncasecmp(dcommand, "encoded ", 8)) { encoded++; command = dcommand + 8; } else { command = dcommand; } if ((arg = strchr(command, ' '))) { *arg++ = '\0'; } SWITCH_STANDARD_STREAM(mystream); switch_api_execute(command, arg, NULL, &mystream); if (mystream.data) { if (encoded) { elen = (int) strlen(mystream.data) * 3; edata = malloc(elen); assert(edata != NULL); memset(edata, 0, elen); switch_url_encode(mystream.data, edata, elen); send = edata; } else { send = mystream.data; } } stream->write_function(stream, "\n" " \n" " %s\n" " \n" "\n", send ? send : "ERROR" ); switch_safe_free(mystream.data); switch_safe_free(edata); free(dcommand); } return SWITCH_STATUS_SUCCESS; } SWITCH_STANDARD_API(sched_api_function) { char *tm = NULL, *dcmd, *group; time_t when; assert(cmd != NULL); tm = strdup(cmd); assert(tm != NULL); if ((group = strchr(tm, ' '))) { uint32_t id; *group++ = '\0'; if ((dcmd = strchr(group, ' '))) { *dcmd++ = '\0'; if (*tm == '+') { when = time(NULL) + atol(tm + 1); } else { when = atol(tm); } id = switch_scheduler_add_task(when, sch_api_callback, (char *) __SWITCH_FUNC__, group, 0, strdup(dcmd), SSHF_FREE_ARG); stream->write_function(stream, "ADDED: %u\n", id); goto good; } } stream->write_function(stream, "Invalid syntax\n"); good: switch_safe_free(tm); return SWITCH_STATUS_SUCCESS; } struct holder { switch_stream_handle_t *stream; char *http; char *delim; uint32_t count; int print_title; switch_xml_t xml; int rows; }; static int show_as_xml_callback(void *pArg, int argc, char **argv, char **columnNames) { struct holder *holder = (struct holder *) pArg; switch_xml_t row, field; int x, f_off = 0; char id[50]; if (holder->count == 0) { if (!(holder->xml = switch_xml_new("result"))) { return -1; } } if (!(row = switch_xml_add_child_d(holder->xml, "row", holder->rows++))) { return -1; } snprintf(id, sizeof(id), "%d", holder->rows); switch_xml_set_attr_d(row, "row_id", id); for(x = 0; x < argc; x++) { char *name = columnNames[x]; char *val = switch_str_nil(argv[x]); if (!name) { name = "undefined"; } if ((field = switch_xml_add_child_d(row, name, f_off++))) { switch_xml_set_txt_d(field, val); } else { return -1; } } holder->count++; return 0; } static int show_callback(void *pArg, int argc, char **argv, char **columnNames) { struct holder *holder = (struct holder *) pArg; int x; if (holder->print_title && holder->count == 0) { if (holder->http) { holder->stream->write_function(holder->stream, "\n"); } for (x = 0; x < argc; x++) { char *name = columnNames[x]; if (!name) { name = "undefined"; } if (holder->http) { holder->stream->write_function(holder->stream, ""); holder->stream->write_function(holder->stream, "%s%s", name, x == (argc - 1) ? "\n" : ""); } else { holder->stream->write_function(holder->stream, "%s%s", name, x == (argc - 1) ? "\n" : holder->delim); } } } if (holder->http) { holder->stream->write_function(holder->stream, "", holder->count % 2 == 0 ? "eeeeee" : "ffffff"); } for (x = 0; x < argc; x++) { char *val = switch_str_nil(argv[x]); if (holder->http) { holder->stream->write_function(holder->stream, ""); holder->stream->write_function(holder->stream, "%s%s", val, x == (argc - 1) ? "\n" : ""); } else { holder->stream->write_function(holder->stream, "%s%s", val, x == (argc - 1) ? "\n" : holder->delim); } } holder->count++; return 0; } SWITCH_STANDARD_API(show_function) { char sql[1024]; char *errmsg; switch_core_db_t *db = switch_core_db_handle(); struct holder holder = { 0 }; int help = 0; char *mydata = NULL, *argv[6] = {0}; int argc; char *command = NULL, *as = NULL; if (session) { return SWITCH_STATUS_FALSE; } if (cmd && (mydata = strdup(cmd))) { argc = switch_separate_string(mydata, ' ', argv, (sizeof(argv) / sizeof(argv[0]))); command = argv[0]; if (argv[2] && !strcasecmp(argv[1], "as")) { as = argv[2]; } } if (!as && stream->event) { holder.http = switch_event_get_header(stream->event, "http-host"); } holder.print_title = 1; // If you changes the field qty or order of any of these select // statmements, you must also change show_callback and friends to match! if (!command) { stream->write_function(stream, "USAGE: %s\n", show_api_interface.syntax); return SWITCH_STATUS_SUCCESS; } else if (!strcmp(command, "codec") || !strcmp(command, "dialplan") || !strcmp(command, "file") || !strcmp(command, "timer")) { sprintf(sql, "select type, name from interfaces where type = '%s'", command); } else if (!strcmp(command, "tasks")) { sprintf(sql, "select * from %s", command); } else if (!strcmp(command, "application") || !strcmp(command, "api")) { sprintf(sql, "select name, description, syntax from interfaces where type = '%s' and description != ''", command); } else if (!strcmp(command, "calls")) { sprintf(sql, "select * from calls"); } else if (!strcmp(command, "channels")) { sprintf(sql, "select * from channels"); } else if (!strncasecmp(command, "help", 4)) { char *cmdname = NULL; help = 1; holder.print_title = 0; if ((cmdname = strchr(command, ' ')) != 0) { *cmdname++ = '\0'; snprintf(sql, sizeof(sql) - 1, "select name, syntax, description from interfaces where type = 'api' and name = '%s'", cmdname); } else { snprintf(sql, sizeof(sql) - 1, "select name, syntax, description from interfaces where type = 'api'"); } } else { stream->write_function(stream, "USAGE: %s\n", show_api_interface.syntax); return SWITCH_STATUS_SUCCESS; } holder.stream = stream; holder.count = 0; if (holder.http) { holder.stream->write_function(holder.stream, "\n"); } if (!as) { as = "delim"; holder.delim = ","; } if (!strcasecmp(as, "delim") || !strcasecmp(as, "csv")) { if (switch_strlen_zero(holder.delim)) { if (!(holder.delim = argv[3])) { holder.delim = ","; } } switch_core_db_exec(db, sql, show_callback, &holder, &errmsg); if (holder.http) { holder.stream->write_function(holder.stream, "
"); } if (errmsg) { stream->write_function(stream, "SQL ERR [%s]\n", errmsg); switch_core_db_free(errmsg); errmsg = NULL; } else if (help) { if (holder.count == 0) stream->write_function(stream, "No such command.\n"); } else { stream->write_function(stream, "\n%u total.\n", holder.count); } } else if (!strcasecmp(as, "xml")) { switch_core_db_exec(db, sql, show_as_xml_callback, &holder, &errmsg); if (holder.xml) { char count[50]; char *xmlstr; snprintf(count, sizeof(count), "%d", holder.count); switch_xml_set_attr_d(holder.xml, "row_count", count); xmlstr = switch_xml_toxml(holder.xml); if (xmlstr) { holder.stream->write_function(holder.stream, "%s", xmlstr); free(xmlstr); } else { holder.stream->write_function(holder.stream, "\n"); } } else { holder.stream->write_function(holder.stream, "\n"); } } else { holder.stream->write_function(holder.stream, "Cannot find format %s\n", as); } switch_safe_free(mydata); switch_core_db_close(db); return SWITCH_STATUS_SUCCESS; } SWITCH_STANDARD_API(version_function) { char version_string[1024]; snprintf(version_string, sizeof(version_string) - 1, "FreeSwitch Version %s\n", SWITCH_VERSION_FULL); stream->write_function(stream, version_string); return SWITCH_STATUS_SUCCESS; } SWITCH_STANDARD_API(help_function) { char showcmd[1024]; int all = 0; if (switch_strlen_zero(cmd)) { sprintf(showcmd, "help"); all = 1; } else { snprintf(showcmd, sizeof(showcmd) - 1, "help %s", cmd); } if (all) { stream->write_function(stream, "\nValid Commands:\n\n"); } show_function(showcmd, session, stream); return SWITCH_STATUS_SUCCESS; } static switch_api_interface_t xml_wrap_api_interface = { /*.interface_name */ "xml_wrap", /*.desc */ "Wrap another api command in xml", /*.function */ xml_wrap_api_function, /*.syntax */ " ", /*.next */ NULL }; static switch_api_interface_t sched_del_api_interface = { /*.interface_name */ "sched_del", /*.desc */ "Delete a Scheduled task", /*.function */ sched_del_function, /*.syntax */ "|", /*.next */ &xml_wrap_api_interface }; static switch_api_interface_t sched_api_api_interface = { /*.interface_name */ "sched_api", /*.desc */ "Schedule an api command", /*.function */ sched_api_function, /*.syntax */ "[+]