2005-11-19 20:07:43 +00:00
|
|
|
/*
|
|
|
|
* FreeSWITCH Modular Media Switching Software Library / Soft-Switch Application
|
2009-02-13 23:37:37 +00:00
|
|
|
* Copyright (C) 2005-2009, Anthony Minessale II <anthm@freeswitch.org>
|
2005-11-19 20:07:43 +00:00
|
|
|
*
|
|
|
|
* 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
|
2009-02-04 21:20:54 +00:00
|
|
|
* Anthony Minessale II <anthm@freeswitch.org>
|
2005-11-19 20:07:43 +00:00
|
|
|
* Portions created by the Initial Developer are Copyright (C)
|
|
|
|
* the Initial Developer. All Rights Reserved.
|
|
|
|
*
|
|
|
|
* Contributor(s):
|
|
|
|
*
|
2009-02-04 21:20:54 +00:00
|
|
|
* Anthony Minessale II <anthm@freeswitch.org>
|
2005-11-19 20:07:43 +00:00
|
|
|
*
|
|
|
|
*
|
|
|
|
* switch_console.c -- Simple Console
|
|
|
|
*
|
|
|
|
*/
|
2008-01-27 17:36:53 +00:00
|
|
|
|
2005-12-14 21:29:46 +00:00
|
|
|
#include <switch.h>
|
2006-04-29 06:05:03 +00:00
|
|
|
#include <switch_console.h>
|
2006-05-15 15:00:10 +00:00
|
|
|
#include <switch_version.h>
|
2006-09-07 03:58:01 +00:00
|
|
|
#define CMD_BUFLEN 1024;
|
2005-11-19 20:07:43 +00:00
|
|
|
|
2008-03-08 04:16:22 +00:00
|
|
|
#ifdef SWITCH_HAVE_LIBEDIT
|
2008-03-03 19:02:36 +00:00
|
|
|
/*
|
|
|
|
* store a strdup() of the string configured in XML
|
|
|
|
* bound to each of the 12 function key
|
|
|
|
*/
|
|
|
|
static char *console_fnkeys[12];
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Load from console.conf XML file the section:
|
|
|
|
* <keybindings>
|
|
|
|
* <key name="1" value="show calls"/>
|
|
|
|
* </keybindings>
|
|
|
|
*/
|
|
|
|
static switch_status_t console_xml_config(void)
|
|
|
|
{
|
|
|
|
char *cf = "switch.conf";
|
|
|
|
switch_xml_t cfg, xml, settings, param;
|
|
|
|
|
2008-05-27 04:30:03 +00:00
|
|
|
/* clear the keybind array */
|
|
|
|
int i;
|
2008-03-03 19:02:36 +00:00
|
|
|
|
2008-05-27 04:30:03 +00:00
|
|
|
for (i = 0; i < 12; i++) {
|
|
|
|
console_fnkeys[i] = NULL;
|
|
|
|
}
|
2008-03-03 19:02:36 +00:00
|
|
|
|
|
|
|
if (!(xml = switch_xml_open_cfg(cf, &cfg, NULL))) {
|
2008-10-11 05:44:11 +00:00
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_ERROR, "Open of %s failed\n", cf);
|
2008-03-03 19:02:36 +00:00
|
|
|
return SWITCH_STATUS_TERM;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((settings = switch_xml_child(cfg, "cli-keybindings"))) {
|
|
|
|
for (param = switch_xml_child(settings, "key"); param; param = param->next) {
|
|
|
|
char *var = (char *) switch_xml_attr_soft(param, "name");
|
|
|
|
char *val = (char *) switch_xml_attr_soft(param, "value");
|
2008-05-27 04:30:03 +00:00
|
|
|
int i = atoi(var);
|
|
|
|
if ((i < 1) || (i > 12)) {
|
2008-10-11 05:44:11 +00:00
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_ERROR, "Keybind %s is invalid, range is from 1 to 12\n", var);
|
2008-05-27 04:30:03 +00:00
|
|
|
} else {
|
2008-09-02 10:46:44 +00:00
|
|
|
/* Add the command to the fnkey array */
|
2008-05-27 04:30:03 +00:00
|
|
|
console_fnkeys[i - 1] = switch_core_permanent_strdup(val);
|
|
|
|
}
|
2008-03-03 19:02:36 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
switch_xml_free(xml);
|
|
|
|
|
|
|
|
return SWITCH_STATUS_SUCCESS;
|
|
|
|
}
|
2008-03-08 04:16:22 +00:00
|
|
|
#endif
|
2008-03-03 19:02:36 +00:00
|
|
|
|
2007-11-28 22:27:33 +00:00
|
|
|
SWITCH_DECLARE_NONSTD(switch_status_t) switch_console_stream_raw_write(switch_stream_handle_t *handle, uint8_t *data, switch_size_t datalen)
|
2007-11-28 19:56:25 +00:00
|
|
|
{
|
2008-09-19 15:46:53 +00:00
|
|
|
switch_size_t need = handle->data_len + datalen;
|
|
|
|
|
|
|
|
if (need >= handle->data_size) {
|
|
|
|
void *new_data;
|
|
|
|
need += handle->alloc_chunk;
|
|
|
|
|
|
|
|
if (!(new_data = realloc(handle->data, need))) {
|
|
|
|
return SWITCH_STATUS_MEMERR;
|
2007-11-29 15:38:00 +00:00
|
|
|
}
|
2008-09-19 15:46:53 +00:00
|
|
|
|
|
|
|
handle->data = new_data;
|
|
|
|
handle->data_size = need;
|
2007-11-28 19:56:25 +00:00
|
|
|
}
|
|
|
|
|
2008-09-19 15:46:53 +00:00
|
|
|
memcpy((uint8_t *) (handle->data) + handle->data_len, data, datalen);
|
|
|
|
handle->data_len += datalen;
|
|
|
|
handle->end = (uint8_t *) (handle->data) + handle->data_len;
|
2008-09-19 17:05:44 +00:00
|
|
|
*(uint8_t *)handle->end = '\0';
|
2008-09-19 15:46:53 +00:00
|
|
|
|
|
|
|
return SWITCH_STATUS_SUCCESS;
|
2007-11-28 19:56:25 +00:00
|
|
|
}
|
2006-05-10 15:47:54 +00:00
|
|
|
|
2007-11-28 22:27:33 +00:00
|
|
|
SWITCH_DECLARE_NONSTD(switch_status_t) switch_console_stream_write(switch_stream_handle_t *handle, const char *fmt, ...)
|
2006-05-10 15:47:54 +00:00
|
|
|
{
|
|
|
|
va_list ap;
|
|
|
|
char *buf = handle->data;
|
|
|
|
char *end = handle->end;
|
|
|
|
int ret = 0;
|
2007-03-23 22:44:28 +00:00
|
|
|
char *data = NULL;
|
2006-05-10 15:47:54 +00:00
|
|
|
|
|
|
|
if (handle->data_len >= handle->data_size) {
|
|
|
|
return SWITCH_STATUS_FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
va_start(ap, fmt);
|
2009-12-10 20:51:13 +00:00
|
|
|
//ret = switch_vasprintf(&data, fmt, ap);
|
|
|
|
if (!(data = switch_vmprintf(fmt, ap))) {
|
|
|
|
ret = -1;
|
|
|
|
}
|
2006-05-10 15:47:54 +00:00
|
|
|
va_end(ap);
|
2007-03-29 22:31:56 +00:00
|
|
|
|
2006-05-10 15:47:54 +00:00
|
|
|
if (data) {
|
2006-09-07 03:58:01 +00:00
|
|
|
switch_size_t remaining = handle->data_size - handle->data_len;
|
|
|
|
switch_size_t need = strlen(data) + 1;
|
2007-03-29 22:31:56 +00:00
|
|
|
|
2006-09-07 03:58:01 +00:00
|
|
|
if ((remaining < need) && handle->alloc_len) {
|
|
|
|
switch_size_t new_len;
|
2007-01-06 16:40:46 +00:00
|
|
|
void *new_data;
|
2006-05-10 15:47:54 +00:00
|
|
|
|
2007-01-06 16:40:46 +00:00
|
|
|
new_len = handle->data_size + need + handle->alloc_chunk;
|
|
|
|
if ((new_data = realloc(handle->data, new_len))) {
|
2006-09-07 03:58:01 +00:00
|
|
|
handle->data_size = handle->alloc_len = new_len;
|
2007-03-29 22:31:56 +00:00
|
|
|
handle->data = new_data;
|
2006-09-07 03:58:01 +00:00
|
|
|
buf = handle->data;
|
|
|
|
remaining = handle->data_size - handle->data_len;
|
2007-03-29 22:31:56 +00:00
|
|
|
handle->end = (uint8_t *) (handle->data) + handle->data_len;
|
2006-09-07 03:58:01 +00:00
|
|
|
end = handle->end;
|
|
|
|
} else {
|
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_CRIT, "Memory Error!\n");
|
|
|
|
free(data);
|
|
|
|
return SWITCH_STATUS_FALSE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (remaining < need) {
|
2006-05-10 15:47:54 +00:00
|
|
|
ret = -1;
|
|
|
|
} else {
|
|
|
|
ret = 0;
|
2007-12-12 21:53:32 +00:00
|
|
|
switch_snprintf(end, remaining, "%s", data);
|
2006-05-10 15:47:54 +00:00
|
|
|
handle->data_len = strlen(buf);
|
2007-03-29 22:31:56 +00:00
|
|
|
handle->end = (uint8_t *) (handle->data) + handle->data_len;
|
2006-05-10 15:47:54 +00:00
|
|
|
}
|
|
|
|
free(data);
|
|
|
|
}
|
2007-03-29 22:31:56 +00:00
|
|
|
|
2006-05-10 15:47:54 +00:00
|
|
|
return ret ? SWITCH_STATUS_FALSE : SWITCH_STATUS_SUCCESS;
|
|
|
|
}
|
|
|
|
|
2009-06-20 03:32:29 +00:00
|
|
|
SWITCH_DECLARE(switch_status_t) switch_stream_write_file_contents(switch_stream_handle_t *stream, const char *path)
|
|
|
|
{
|
|
|
|
char *dpath = NULL;
|
|
|
|
int fd;
|
|
|
|
switch_status_t status = SWITCH_STATUS_FALSE;
|
|
|
|
|
|
|
|
if (!switch_is_file_path(path)) {
|
|
|
|
dpath = switch_mprintf("%s%s%s", SWITCH_GLOBAL_dirs.conf_dir, SWITCH_PATH_SEPARATOR, path);
|
|
|
|
path = dpath;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((fd = open(path, O_RDONLY)) > -1) {
|
|
|
|
char buf[2048] = { 0 };
|
|
|
|
while (switch_fd_read_line(fd, buf, sizeof(buf))) {
|
|
|
|
stream->write_function(stream, "%s", buf);
|
|
|
|
}
|
|
|
|
close(fd);
|
|
|
|
status = SWITCH_STATUS_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
switch_safe_free(dpath);
|
|
|
|
return status;
|
|
|
|
}
|
|
|
|
|
2008-04-23 17:38:13 +00:00
|
|
|
static int alias_callback(void *pArg, int argc, char **argv, char **columnNames)
|
2005-11-19 20:07:43 +00:00
|
|
|
{
|
2008-04-23 17:38:13 +00:00
|
|
|
char **r = (char **) pArg;
|
|
|
|
*r = strdup(argv[0]);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
char *expand_alias(char *cmd, char *arg)
|
|
|
|
{
|
|
|
|
char *errmsg = NULL;
|
|
|
|
char *r = NULL;
|
2008-08-11 17:47:06 +00:00
|
|
|
char *sql = NULL;
|
2008-04-23 17:38:13 +00:00
|
|
|
char *exp = NULL;
|
2009-11-17 00:12:54 +00:00
|
|
|
switch_cache_db_handle_t *db = NULL;
|
2008-04-23 22:15:07 +00:00
|
|
|
int full = 0;
|
2008-05-27 04:30:03 +00:00
|
|
|
|
2009-11-17 00:12:54 +00:00
|
|
|
|
|
|
|
switch_core_db_handle(&db);
|
|
|
|
|
2009-12-10 20:51:13 +00:00
|
|
|
|
|
|
|
if (db->type == SCDB_TYPE_CORE_DB) {
|
|
|
|
sql = switch_mprintf("select command from aliases where alias='%q'", cmd);
|
|
|
|
} else {
|
|
|
|
sql = switch_mprintf("select command from aliases where alias='%w'", cmd);
|
|
|
|
}
|
2008-04-23 22:15:07 +00:00
|
|
|
|
2009-11-17 00:12:54 +00:00
|
|
|
switch_cache_db_execute_sql_callback(db, sql, alias_callback, &r, &errmsg);
|
2008-04-23 17:38:13 +00:00
|
|
|
|
|
|
|
if (errmsg) {
|
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_ERROR, "error [%s][%s]\n", sql, errmsg);
|
|
|
|
free(errmsg);
|
|
|
|
}
|
|
|
|
|
2008-08-11 17:47:06 +00:00
|
|
|
switch_safe_free(sql);
|
|
|
|
|
2008-04-23 22:15:07 +00:00
|
|
|
if (!r) {
|
2009-12-10 20:51:13 +00:00
|
|
|
if (db->type == SCDB_TYPE_CORE_DB) {
|
|
|
|
sql = switch_mprintf("select command from aliases where alias='%q %q'", cmd, arg);
|
|
|
|
} else {
|
|
|
|
sql = switch_mprintf("select command from aliases where alias='%w %w'", cmd, arg);
|
|
|
|
}
|
2008-05-27 04:30:03 +00:00
|
|
|
|
2009-11-17 00:12:54 +00:00
|
|
|
switch_cache_db_execute_sql_callback(db, sql, alias_callback, &r, &errmsg);
|
2008-05-27 04:30:03 +00:00
|
|
|
|
2008-04-23 22:15:07 +00:00
|
|
|
if (errmsg) {
|
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_ERROR, "error [%s][%s]\n", sql, errmsg);
|
|
|
|
free(errmsg);
|
|
|
|
}
|
|
|
|
if (r) {
|
|
|
|
full++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2008-08-11 17:47:06 +00:00
|
|
|
switch_safe_free(sql);
|
2008-04-23 22:15:07 +00:00
|
|
|
|
2008-04-23 17:38:13 +00:00
|
|
|
if (r) {
|
2008-04-23 22:15:07 +00:00
|
|
|
if (arg && !full) {
|
2008-04-23 17:38:13 +00:00
|
|
|
exp = switch_mprintf("%s %s", r, arg);
|
|
|
|
free(r);
|
|
|
|
} else {
|
|
|
|
exp = r;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
exp = cmd;
|
|
|
|
}
|
|
|
|
|
2009-11-17 00:12:54 +00:00
|
|
|
switch_cache_db_release_db_handle(&db);
|
|
|
|
|
2008-04-23 17:38:13 +00:00
|
|
|
return exp;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int switch_console_process(char *cmd, int rec)
|
|
|
|
{
|
|
|
|
char *arg = NULL, *alias = NULL;
|
2007-03-29 22:31:56 +00:00
|
|
|
switch_stream_handle_t stream = { 0 };
|
2006-03-29 19:11:20 +00:00
|
|
|
|
2005-11-19 20:07:43 +00:00
|
|
|
if (!strcmp(cmd, "shutdown") || !strcmp(cmd, "...")) {
|
2006-04-11 21:13:44 +00:00
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_CONSOLE, "Bye!\n");
|
2005-11-19 20:07:43 +00:00
|
|
|
return 0;
|
|
|
|
}
|
2006-01-06 17:44:19 +00:00
|
|
|
if (!strcmp(cmd, "version")) {
|
2008-09-01 04:46:56 +00:00
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG_CLEAN, SWITCH_LOG_CONSOLE, "FreeSWITCH Version %s\n", SWITCH_VERSION_FULL);
|
2006-01-06 17:44:19 +00:00
|
|
|
return 1;
|
|
|
|
}
|
2007-03-29 22:31:56 +00:00
|
|
|
if ((arg = strchr(cmd, '\r')) != 0 || (arg = strchr(cmd, '\n')) != 0) {
|
2005-12-23 16:54:49 +00:00
|
|
|
*arg = '\0';
|
|
|
|
arg = NULL;
|
|
|
|
}
|
2006-02-20 00:23:25 +00:00
|
|
|
if ((arg = strchr(cmd, ' ')) != 0) {
|
2005-12-23 16:54:49 +00:00
|
|
|
*arg++ = '\0';
|
|
|
|
}
|
2007-03-29 22:31:56 +00:00
|
|
|
|
2008-04-23 17:38:13 +00:00
|
|
|
if (!rec && (alias = expand_alias(cmd, arg)) && alias != cmd) {
|
|
|
|
int r = switch_console_process(alias, ++rec);
|
|
|
|
free(alias);
|
|
|
|
return r;
|
|
|
|
}
|
|
|
|
|
2006-09-07 03:58:01 +00:00
|
|
|
SWITCH_STANDARD_STREAM(stream);
|
|
|
|
if (stream.data) {
|
2008-12-29 23:38:59 +00:00
|
|
|
FILE *handle = switch_core_get_console();
|
|
|
|
|
2006-09-07 03:58:01 +00:00
|
|
|
if (switch_api_execute(cmd, arg, NULL, &stream) == SWITCH_STATUS_SUCCESS) {
|
2008-12-29 23:38:59 +00:00
|
|
|
if (handle) {
|
|
|
|
fprintf(handle, "API CALL [%s(%s)] output:\n%s\n", cmd, arg ? arg : "", (char *) stream.data);
|
|
|
|
fflush(handle);
|
|
|
|
}
|
2006-09-07 03:58:01 +00:00
|
|
|
} else {
|
2008-12-29 23:38:59 +00:00
|
|
|
if (handle) {
|
|
|
|
fprintf(handle, "Unknown Command: %s\n", cmd);
|
|
|
|
fflush(handle);
|
|
|
|
}
|
2006-09-07 03:58:01 +00:00
|
|
|
}
|
|
|
|
free(stream.data);
|
2005-12-23 02:55:25 +00:00
|
|
|
} else {
|
2006-09-07 03:58:01 +00:00
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_ERROR, "Memory Error!\n");
|
2005-12-23 02:55:25 +00:00
|
|
|
}
|
2006-09-07 03:58:01 +00:00
|
|
|
|
2005-11-19 20:07:43 +00:00
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2007-03-30 00:13:31 +00:00
|
|
|
SWITCH_DECLARE(void) switch_console_printf(switch_text_channel_t channel, const char *file, const char *func, int line, const char *fmt, ...)
|
2005-11-19 20:07:43 +00:00
|
|
|
{
|
2006-02-13 18:34:39 +00:00
|
|
|
char *data = NULL;
|
2005-11-19 20:07:43 +00:00
|
|
|
int ret = 0;
|
|
|
|
va_list ap;
|
2007-12-11 03:32:21 +00:00
|
|
|
FILE *handle = switch_core_data_channel(channel);
|
2007-02-13 21:03:06 +00:00
|
|
|
const char *filep = switch_cut_path(file);
|
2007-12-11 03:32:21 +00:00
|
|
|
char date[80] = "";
|
|
|
|
switch_size_t retsize;
|
|
|
|
switch_time_exp_t tm;
|
|
|
|
switch_event_t *event;
|
2005-11-19 20:07:43 +00:00
|
|
|
|
|
|
|
va_start(ap, fmt);
|
2007-03-09 20:44:13 +00:00
|
|
|
ret = switch_vasprintf(&data, fmt, ap);
|
2005-11-19 20:07:43 +00:00
|
|
|
va_end(ap);
|
2007-12-11 03:32:21 +00:00
|
|
|
|
2005-11-19 20:07:43 +00:00
|
|
|
if (ret == -1) {
|
|
|
|
fprintf(stderr, "Memory Error\n");
|
2007-12-11 03:32:21 +00:00
|
|
|
goto done;
|
2008-05-27 04:30:03 +00:00
|
|
|
}
|
2005-11-19 20:07:43 +00:00
|
|
|
|
2007-12-11 03:32:21 +00:00
|
|
|
if (channel == SWITCH_CHANNEL_ID_LOG_CLEAN) {
|
|
|
|
fprintf(handle, "%s", data);
|
|
|
|
goto done;
|
|
|
|
}
|
2005-12-21 22:25:22 +00:00
|
|
|
|
2009-01-25 21:23:07 +00:00
|
|
|
switch_time_exp_lt(&tm, switch_micro_time_now());
|
2008-10-12 21:51:51 +00:00
|
|
|
switch_strftime_nocheck(date, &retsize, sizeof(date), "%Y-%m-%d %T", &tm);
|
2006-01-03 01:17:59 +00:00
|
|
|
|
2007-12-11 03:32:21 +00:00
|
|
|
if (channel == SWITCH_CHANNEL_ID_LOG) {
|
|
|
|
fprintf(handle, "[%d] %s %s:%d %s() %s", (int) getpid(), date, filep, line, func, data);
|
|
|
|
goto done;
|
2008-05-27 04:30:03 +00:00
|
|
|
}
|
2007-12-11 03:32:21 +00:00
|
|
|
|
|
|
|
if (channel == SWITCH_CHANNEL_ID_EVENT &&
|
2008-05-27 04:30:03 +00:00
|
|
|
switch_event_running() == SWITCH_STATUS_SUCCESS && switch_event_create(&event, SWITCH_EVENT_LOG) == SWITCH_STATUS_SUCCESS) {
|
2007-12-11 03:32:21 +00:00
|
|
|
|
2008-08-16 02:17:09 +00:00
|
|
|
switch_event_add_header_string(event, SWITCH_STACK_BOTTOM, "Log-Data", data);
|
|
|
|
switch_event_add_header_string(event, SWITCH_STACK_BOTTOM, "Log-File", filep);
|
|
|
|
switch_event_add_header_string(event, SWITCH_STACK_BOTTOM, "Log-Function", func);
|
2007-12-11 03:32:21 +00:00
|
|
|
switch_event_add_header(event, SWITCH_STACK_BOTTOM, "Log-Line", "%d", line);
|
|
|
|
switch_event_fire(&event);
|
2005-11-19 20:07:43 +00:00
|
|
|
}
|
2007-12-11 03:32:21 +00:00
|
|
|
|
2008-05-27 04:30:03 +00:00
|
|
|
done:
|
2007-03-29 22:31:56 +00:00
|
|
|
if (data) {
|
2006-02-13 18:34:39 +00:00
|
|
|
free(data);
|
|
|
|
}
|
2005-11-19 20:07:43 +00:00
|
|
|
fflush(handle);
|
|
|
|
}
|
|
|
|
|
2007-04-30 20:37:41 +00:00
|
|
|
static char hostname[256] = "";
|
2007-05-14 16:20:57 +00:00
|
|
|
static int32_t running = 1;
|
2007-04-30 20:37:41 +00:00
|
|
|
|
|
|
|
#ifdef SWITCH_HAVE_LIBEDIT
|
|
|
|
#include <histedit.h>
|
|
|
|
static char prompt_str[512] = "";
|
|
|
|
|
2008-03-03 19:02:36 +00:00
|
|
|
/*
|
|
|
|
* If a fnkey is configured then process the command
|
|
|
|
*/
|
2008-05-27 04:30:03 +00:00
|
|
|
static unsigned char console_fnkey_pressed(int i)
|
|
|
|
{
|
2008-03-05 20:46:21 +00:00
|
|
|
char *c, *cmd;
|
2008-03-03 19:02:36 +00:00
|
|
|
|
2008-05-27 04:30:03 +00:00
|
|
|
assert((i > 0) && (i <= 12));
|
2008-03-03 19:02:36 +00:00
|
|
|
|
2008-05-27 04:30:03 +00:00
|
|
|
c = console_fnkeys[i - 1];
|
2008-03-03 19:02:36 +00:00
|
|
|
|
2008-09-02 10:46:44 +00:00
|
|
|
/* This new line is necessary to avoid output to begin after the ">" of the CLI's prompt */
|
2008-05-27 04:30:03 +00:00
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG_CLEAN, SWITCH_LOG_CONSOLE, "\n");
|
|
|
|
|
|
|
|
if (c == NULL) {
|
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_CONSOLE, "FUNCTION KEY F%d IS NOT BOUND, please edit switch.conf XML file\n", i);
|
|
|
|
return CC_REDISPLAY;
|
|
|
|
}
|
2008-03-03 19:02:36 +00:00
|
|
|
|
2008-03-05 20:46:21 +00:00
|
|
|
cmd = strdup(c);
|
2008-04-23 17:38:13 +00:00
|
|
|
switch_console_process(cmd, 0);
|
2008-03-05 20:46:21 +00:00
|
|
|
free(cmd);
|
2008-03-03 19:02:36 +00:00
|
|
|
|
2008-05-27 04:30:03 +00:00
|
|
|
return CC_REDISPLAY;
|
2008-03-03 19:02:36 +00:00
|
|
|
}
|
|
|
|
|
2008-05-27 04:30:03 +00:00
|
|
|
static unsigned char console_f1key(EditLine * el, int ch)
|
|
|
|
{
|
|
|
|
return console_fnkey_pressed(1);
|
2008-03-03 19:02:36 +00:00
|
|
|
}
|
2008-05-27 04:30:03 +00:00
|
|
|
static unsigned char console_f2key(EditLine * el, int ch)
|
|
|
|
{
|
|
|
|
return console_fnkey_pressed(2);
|
2008-03-03 19:02:36 +00:00
|
|
|
}
|
2008-05-27 04:30:03 +00:00
|
|
|
static unsigned char console_f3key(EditLine * el, int ch)
|
|
|
|
{
|
|
|
|
return console_fnkey_pressed(3);
|
2008-03-03 19:02:36 +00:00
|
|
|
}
|
2008-05-27 04:30:03 +00:00
|
|
|
static unsigned char console_f4key(EditLine * el, int ch)
|
|
|
|
{
|
|
|
|
return console_fnkey_pressed(4);
|
2008-03-03 19:02:36 +00:00
|
|
|
}
|
2008-05-27 04:30:03 +00:00
|
|
|
static unsigned char console_f5key(EditLine * el, int ch)
|
|
|
|
{
|
|
|
|
return console_fnkey_pressed(5);
|
2008-03-03 19:02:36 +00:00
|
|
|
}
|
2008-05-27 04:30:03 +00:00
|
|
|
static unsigned char console_f6key(EditLine * el, int ch)
|
|
|
|
{
|
|
|
|
return console_fnkey_pressed(6);
|
2008-03-03 19:02:36 +00:00
|
|
|
}
|
2008-05-27 04:30:03 +00:00
|
|
|
static unsigned char console_f7key(EditLine * el, int ch)
|
|
|
|
{
|
|
|
|
return console_fnkey_pressed(7);
|
2008-03-03 19:02:36 +00:00
|
|
|
}
|
2008-05-27 04:30:03 +00:00
|
|
|
static unsigned char console_f8key(EditLine * el, int ch)
|
|
|
|
{
|
|
|
|
return console_fnkey_pressed(8);
|
2008-03-03 19:02:36 +00:00
|
|
|
}
|
2008-05-27 04:30:03 +00:00
|
|
|
static unsigned char console_f9key(EditLine * el, int ch)
|
|
|
|
{
|
|
|
|
return console_fnkey_pressed(9);
|
2008-03-03 19:02:36 +00:00
|
|
|
}
|
2008-05-27 04:30:03 +00:00
|
|
|
static unsigned char console_f10key(EditLine * el, int ch)
|
|
|
|
{
|
|
|
|
return console_fnkey_pressed(10);
|
2008-03-03 19:02:36 +00:00
|
|
|
}
|
2008-05-27 04:30:03 +00:00
|
|
|
static unsigned char console_f11key(EditLine * el, int ch)
|
|
|
|
{
|
|
|
|
return console_fnkey_pressed(11);
|
2008-03-03 19:02:36 +00:00
|
|
|
}
|
2008-05-27 04:30:03 +00:00
|
|
|
static unsigned char console_f12key(EditLine * el, int ch)
|
|
|
|
{
|
|
|
|
return console_fnkey_pressed(12);
|
2008-03-03 19:02:36 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-05-27 04:30:03 +00:00
|
|
|
char *prompt(EditLine * e)
|
|
|
|
{
|
2007-08-29 20:38:00 +00:00
|
|
|
if (*prompt_str == '\0') {
|
2007-12-12 21:53:32 +00:00
|
|
|
switch_snprintf(prompt_str, sizeof(prompt_str), "freeswitch@%s> ", hostname);
|
2008-05-27 04:30:03 +00:00
|
|
|
}
|
2007-04-30 20:37:41 +00:00
|
|
|
|
|
|
|
return prompt_str;
|
|
|
|
}
|
|
|
|
|
2007-05-14 16:20:57 +00:00
|
|
|
static EditLine *el;
|
|
|
|
static History *myhistory;
|
|
|
|
static HistEvent ev;
|
|
|
|
static char *hfile = NULL;
|
|
|
|
|
|
|
|
static void *SWITCH_THREAD_FUNC console_thread(switch_thread_t *thread, void *obj)
|
2005-11-19 20:07:43 +00:00
|
|
|
{
|
2007-04-30 20:37:41 +00:00
|
|
|
int count;
|
|
|
|
const char *line;
|
2007-05-14 16:20:57 +00:00
|
|
|
switch_memory_pool_t *pool = (switch_memory_pool_t *) obj;
|
2007-04-30 20:37:41 +00:00
|
|
|
|
|
|
|
while (running) {
|
2008-07-15 15:51:10 +00:00
|
|
|
int32_t arg = 0;
|
2007-04-30 23:56:27 +00:00
|
|
|
|
2008-07-15 15:51:10 +00:00
|
|
|
if (getppid() == 1) {
|
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_WARNING, "We've become an orphan, no more console for us.\n");
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2007-04-30 23:56:27 +00:00
|
|
|
switch_core_session_ctl(SCSC_CHECK_RUNNING, &arg);
|
|
|
|
if (!arg) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2007-04-30 20:37:41 +00:00
|
|
|
line = el_gets(el, &count);
|
2007-04-30 23:56:27 +00:00
|
|
|
|
2007-04-30 20:37:41 +00:00
|
|
|
if (count > 1) {
|
2009-10-23 16:03:42 +00:00
|
|
|
if (!zstr(line)) {
|
2007-04-30 20:37:41 +00:00
|
|
|
char *cmd = strdup(line);
|
|
|
|
char *p;
|
2008-04-22 20:49:39 +00:00
|
|
|
const LineInfo *lf = el_line(el);
|
2008-05-27 04:30:03 +00:00
|
|
|
char *foo = (char *) lf->buffer;
|
2007-04-30 20:37:41 +00:00
|
|
|
if ((p = strrchr(cmd, '\r')) || (p = strrchr(cmd, '\n'))) {
|
|
|
|
*p = '\0';
|
|
|
|
}
|
|
|
|
assert(cmd != NULL);
|
|
|
|
history(myhistory, &ev, H_ENTER, line);
|
2008-04-23 17:38:13 +00:00
|
|
|
running = switch_console_process(cmd, 0);
|
2008-05-27 04:30:03 +00:00
|
|
|
el_deletestr(el, strlen(foo) + 1);
|
2008-04-22 20:49:39 +00:00
|
|
|
memset(foo, 0, strlen(foo));
|
2007-04-30 20:37:41 +00:00
|
|
|
free(cmd);
|
|
|
|
}
|
|
|
|
}
|
2008-11-14 23:31:21 +00:00
|
|
|
switch_cond_next();
|
2007-04-30 20:37:41 +00:00
|
|
|
}
|
2008-05-27 04:30:03 +00:00
|
|
|
|
|
|
|
switch_core_destroy_memory_pool(&pool);
|
2008-07-15 15:51:10 +00:00
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_DEBUG, "Editline thread exiting\n");
|
2007-05-14 16:20:57 +00:00
|
|
|
return NULL;
|
|
|
|
}
|
2007-04-30 20:37:41 +00:00
|
|
|
|
|
|
|
|
2008-04-22 20:49:39 +00:00
|
|
|
struct helper {
|
2009-12-15 00:34:02 +00:00
|
|
|
EditLine *el;
|
2008-04-22 20:49:39 +00:00
|
|
|
int len;
|
|
|
|
int hits;
|
2008-04-23 16:46:24 +00:00
|
|
|
int words;
|
2008-04-22 20:49:39 +00:00
|
|
|
char last[512];
|
2009-12-10 20:51:13 +00:00
|
|
|
char partial[512];
|
2008-04-22 20:49:39 +00:00
|
|
|
FILE *out;
|
2009-12-15 00:34:02 +00:00
|
|
|
switch_stream_handle_t *stream;
|
2008-04-22 20:49:39 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
static int comp_callback(void *pArg, int argc, char **argv, char **columnNames)
|
|
|
|
{
|
|
|
|
struct helper *h = (struct helper *) pArg;
|
2009-12-14 20:10:06 +00:00
|
|
|
char *target = NULL, *str = NULL, *cur = NULL;
|
2009-12-10 20:51:13 +00:00
|
|
|
switch_size_t x, y;
|
|
|
|
int i;
|
|
|
|
|
2009-12-14 20:10:06 +00:00
|
|
|
if (argc > 0) target = argv[0];
|
|
|
|
if (argc > 1) str = argv[1];
|
|
|
|
if (argc > 2) cur = argv[2];
|
|
|
|
|
|
|
|
if (cur) {
|
|
|
|
while (*cur == ' ') cur++;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (zstr(cur)) cur = NULL;
|
|
|
|
if (zstr(str)) str = NULL;
|
2008-04-23 16:46:24 +00:00
|
|
|
|
|
|
|
if (!target) {
|
|
|
|
return -1;
|
|
|
|
}
|
2008-04-22 20:49:39 +00:00
|
|
|
|
2009-12-14 20:10:06 +00:00
|
|
|
if (!zstr(target) && *target == ':' && *(target+1) == ':') {
|
|
|
|
char *r_argv[3] = { 0 }, *r_cols[3] = { 0 };
|
|
|
|
switch_console_callback_match_t *matches;
|
|
|
|
r_cols[0] = "match";
|
|
|
|
if (switch_console_run_complete_func(target, str, cur, &matches) == SWITCH_STATUS_SUCCESS) {
|
|
|
|
switch_console_callback_match_node_t *m;
|
|
|
|
for (m = matches->head; m; m = m->next) {
|
|
|
|
if (!cur || !strncmp(m->val, cur, strlen(cur))) {
|
|
|
|
r_argv[0] = m->val;
|
|
|
|
comp_callback(h, 1, r_argv, r_cols);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
switch_console_free_matches(&matches);
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2009-12-10 20:51:13 +00:00
|
|
|
if (!zstr(target)) {
|
2009-12-15 00:34:02 +00:00
|
|
|
if (h->out) {
|
|
|
|
fprintf(h->out, "[%20s]\t", target);
|
|
|
|
}
|
|
|
|
if (h->stream) {
|
|
|
|
h->stream->write_function(h->stream, "[%20s]\t", target);
|
|
|
|
}
|
|
|
|
|
2009-12-10 20:51:13 +00:00
|
|
|
switch_copy_string(h->last, target, sizeof(h->last));
|
|
|
|
h->hits++;
|
|
|
|
}
|
|
|
|
|
|
|
|
x = strlen(h->last);
|
|
|
|
y = strlen(h->partial);
|
|
|
|
|
|
|
|
if (h->hits > 1) {
|
|
|
|
for(i = 0; i < x && i < y; i++) {
|
|
|
|
if (h->last[i] != h->partial[i]) {
|
|
|
|
h->partial[i] = '\0';
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else if (h->hits == 1) {
|
|
|
|
switch_copy_string(h->partial, target, sizeof(h->last));
|
|
|
|
}
|
2008-04-22 20:49:39 +00:00
|
|
|
|
2009-10-23 16:03:42 +00:00
|
|
|
if (!zstr(target)) {
|
2009-12-10 20:51:13 +00:00
|
|
|
if ((h->hits % 4) == 0) {
|
2009-12-15 00:34:02 +00:00
|
|
|
if (h->out) {
|
|
|
|
fprintf(h->out, "\n");
|
|
|
|
}
|
|
|
|
if (h->stream) {
|
|
|
|
h->stream->write_function(h->stream, "\n");
|
|
|
|
}
|
2008-07-22 22:23:50 +00:00
|
|
|
}
|
2008-04-22 20:49:39 +00:00
|
|
|
}
|
2008-05-27 04:30:03 +00:00
|
|
|
|
2008-04-22 20:49:39 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2009-12-14 20:10:06 +00:00
|
|
|
|
2009-12-15 00:34:02 +00:00
|
|
|
|
|
|
|
struct match_helper {
|
|
|
|
switch_console_callback_match_t *my_matches;
|
|
|
|
};
|
|
|
|
|
|
|
|
static int uuid_callback(void *pArg, int argc, char **argv, char **columnNames)
|
|
|
|
{
|
|
|
|
struct match_helper *h = (struct match_helper *) pArg;
|
|
|
|
|
|
|
|
switch_console_push_match(&h->my_matches, argv[0]);
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
SWITCH_DECLARE(switch_status_t) switch_console_list_uuid(const char *line, const char *cursor, switch_console_callback_match_t **matches)
|
|
|
|
{
|
|
|
|
char *sql;
|
|
|
|
struct match_helper h = { 0 };
|
|
|
|
switch_cache_db_handle_t *db = NULL;
|
|
|
|
switch_status_t status = SWITCH_STATUS_FALSE;
|
|
|
|
char *errmsg;
|
|
|
|
|
|
|
|
switch_core_db_handle(&db);
|
|
|
|
|
|
|
|
if (!zstr(cursor)) {
|
|
|
|
sql = switch_mprintf("select distinct uuid from channels where uuid like '%q%%' and hostname='%q' order by uuid",
|
|
|
|
cursor, switch_core_get_variable("hostname"));
|
|
|
|
} else {
|
|
|
|
sql = switch_mprintf("select distinct uuid from channels where hostname='%q' order by uuid",
|
|
|
|
switch_core_get_variable("hostname"));
|
|
|
|
}
|
|
|
|
|
|
|
|
switch_cache_db_execute_sql_callback(db, sql, uuid_callback, &h, &errmsg);
|
|
|
|
free(sql);
|
|
|
|
|
|
|
|
switch_cache_db_release_db_handle(&db);
|
|
|
|
|
|
|
|
if (h.my_matches) {
|
|
|
|
*matches = h.my_matches;
|
|
|
|
status = SWITCH_STATUS_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
return status;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
SWITCH_DECLARE(unsigned char) switch_console_complete(const char *line, const char *cursor, FILE *console_out, switch_stream_handle_t *stream)
|
2008-04-22 20:49:39 +00:00
|
|
|
{
|
2009-11-17 00:12:54 +00:00
|
|
|
switch_cache_db_handle_t *db = NULL;
|
2009-12-11 05:14:23 +00:00
|
|
|
char *sql = NULL;
|
2009-12-15 00:34:02 +00:00
|
|
|
char *dup = strdup(line);
|
2008-04-22 20:49:39 +00:00
|
|
|
char *buf = dup;
|
|
|
|
char *p, *lp = NULL;
|
|
|
|
char *errmsg = NULL;
|
2009-12-15 00:34:02 +00:00
|
|
|
struct helper h = { el };
|
2008-04-22 20:49:39 +00:00
|
|
|
unsigned char ret = CC_REDISPLAY;
|
2009-12-11 05:14:23 +00:00
|
|
|
int pos = 0;
|
2008-04-22 20:49:39 +00:00
|
|
|
|
2009-11-17 00:12:54 +00:00
|
|
|
switch_core_db_handle(&db);
|
2009-12-11 05:14:23 +00:00
|
|
|
|
2009-12-15 00:34:02 +00:00
|
|
|
if (!zstr(cursor) && !zstr(line)) {
|
|
|
|
pos = (cursor - line);
|
2009-12-11 05:14:23 +00:00
|
|
|
}
|
|
|
|
|
2009-12-15 00:34:02 +00:00
|
|
|
h.out = console_out;
|
|
|
|
h.stream = stream;
|
2008-04-22 20:49:39 +00:00
|
|
|
|
2009-12-11 05:14:23 +00:00
|
|
|
if (pos > 0) {
|
|
|
|
*(buf + pos) = '\0';
|
|
|
|
}
|
|
|
|
|
2008-04-22 20:49:39 +00:00
|
|
|
if ((p = strchr(buf, '\r')) || (p = strchr(buf, '\n'))) {
|
|
|
|
*p = '\0';
|
2008-05-27 04:30:03 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
while (*buf == ' ') {
|
2008-04-22 20:49:39 +00:00
|
|
|
buf++;
|
|
|
|
}
|
|
|
|
|
2008-05-27 04:30:03 +00:00
|
|
|
for (p = buf; p && *p; p++) {
|
2008-04-22 20:49:39 +00:00
|
|
|
if (*p == ' ') {
|
|
|
|
lp = p;
|
2008-04-23 16:46:24 +00:00
|
|
|
h.words++;
|
2008-04-22 20:49:39 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (lp) {
|
|
|
|
buf = lp + 1;
|
|
|
|
}
|
2008-05-27 04:30:03 +00:00
|
|
|
|
2008-04-22 20:49:39 +00:00
|
|
|
h.len = strlen(buf);
|
2009-12-11 05:14:23 +00:00
|
|
|
|
2009-12-15 00:34:02 +00:00
|
|
|
if (h.out) {
|
|
|
|
fprintf(h.out, "\n\n");
|
|
|
|
}
|
|
|
|
|
|
|
|
if (h.stream) {
|
|
|
|
h.stream->write_function(h.stream, "\n\n");
|
|
|
|
}
|
2008-04-22 20:49:39 +00:00
|
|
|
|
2008-04-23 16:46:24 +00:00
|
|
|
if (h.words == 0) {
|
2009-11-17 00:12:54 +00:00
|
|
|
sql = switch_mprintf("select distinct name from interfaces where type='api' and name like '%q%%' and hostname='%q' order by name",
|
|
|
|
buf, switch_core_get_variable("hostname"));
|
2008-04-22 20:49:39 +00:00
|
|
|
}
|
2009-12-14 20:10:06 +00:00
|
|
|
|
2009-12-11 05:14:23 +00:00
|
|
|
if (sql) {
|
|
|
|
switch_cache_db_execute_sql_callback(db, sql, comp_callback, &h, &errmsg);
|
2008-05-27 04:30:03 +00:00
|
|
|
|
2009-12-11 05:14:23 +00:00
|
|
|
if (errmsg) {
|
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_ERROR, "error [%s][%s]\n", sql, errmsg);
|
|
|
|
free(errmsg);
|
|
|
|
ret = CC_ERROR;
|
|
|
|
goto end;
|
|
|
|
}
|
|
|
|
free(sql);
|
|
|
|
sql = NULL;
|
2008-04-22 20:49:39 +00:00
|
|
|
}
|
|
|
|
|
2009-12-11 05:14:23 +00:00
|
|
|
if (h.hits != -1) {
|
2008-04-23 16:46:24 +00:00
|
|
|
char *dupdup = strdup(dup);
|
|
|
|
int x, argc = 0;
|
2008-05-27 04:30:03 +00:00
|
|
|
char *argv[10] = { 0 };
|
2008-04-23 16:46:24 +00:00
|
|
|
switch_stream_handle_t stream = { 0 };
|
|
|
|
SWITCH_STANDARD_STREAM(stream);
|
2009-12-02 20:28:49 +00:00
|
|
|
switch_assert(dupdup);
|
2008-05-27 04:30:03 +00:00
|
|
|
|
2008-04-23 16:46:24 +00:00
|
|
|
argc = switch_separate_string(dupdup, ' ', argv, (sizeof(argv) / sizeof(argv[0])));
|
|
|
|
|
|
|
|
if (h.words == 0) {
|
2009-11-17 00:12:54 +00:00
|
|
|
stream.write_function(&stream, "select distinct a1 from complete where " "a1 not in (select name from interfaces where hostname='%s') %s ",
|
|
|
|
switch_core_get_variable("hostname"), argc ? "and" : "");
|
2008-04-23 16:46:24 +00:00
|
|
|
} else {
|
2009-12-14 20:10:06 +00:00
|
|
|
if (db->type == SCDB_TYPE_CORE_DB) {
|
|
|
|
stream.write_function(&stream, "select distinct a%d,'%q','%q' from complete where ", h.words + 1, switch_str_nil(dup), switch_str_nil(lp));
|
|
|
|
} else {
|
|
|
|
stream.write_function(&stream, "select distinct a%d,'%q','%w' from complete where ", h.words + 1, switch_str_nil(dup), switch_str_nil(lp));
|
|
|
|
}
|
2008-04-23 16:46:24 +00:00
|
|
|
}
|
|
|
|
|
2009-12-10 20:51:13 +00:00
|
|
|
for (x = 0; x < argc && x < 11; x++) {
|
|
|
|
if (h.words + 1 > argc) {
|
|
|
|
if (db->type == SCDB_TYPE_CORE_DB) {
|
2009-12-14 20:10:06 +00:00
|
|
|
stream.write_function(&stream, "(a%d like '::%%' or a%d = '' or a%d = '%q')%q",
|
|
|
|
x + 1, x + 1, x + 1, switch_str_nil(argv[x]), x == argc - 1 ? "" : " and ");
|
2009-12-10 20:51:13 +00:00
|
|
|
} else {
|
2009-12-14 20:10:06 +00:00
|
|
|
stream.write_function(&stream, "(a%d like '::%%' or a%d = '' or a%d = '%w')%w",
|
|
|
|
x + 1, x + 1, x + 1, switch_str_nil(argv[x]), x == argc - 1 ? "" : " and ");
|
2009-12-10 20:51:13 +00:00
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if (db->type == SCDB_TYPE_CORE_DB) {
|
2009-12-14 20:10:06 +00:00
|
|
|
stream.write_function(&stream, "(a%d like '::%%' or a%d = '' or a%d like '%q%%')%q",
|
|
|
|
x + 1, x + 1, x + 1, switch_str_nil(argv[x]), x == argc - 1 ? "" : " and ");
|
2009-12-10 20:51:13 +00:00
|
|
|
} else {
|
2009-12-14 20:10:06 +00:00
|
|
|
stream.write_function(&stream, "(a%d like '::%%' or a%d = '' or a%d like '%w%%')%w",
|
|
|
|
x + 1, x + 1, x + 1, switch_str_nil(argv[x]), x == argc - 1 ? "" : " and ");
|
2009-12-10 20:51:13 +00:00
|
|
|
}
|
|
|
|
}
|
2008-04-23 16:46:24 +00:00
|
|
|
}
|
2008-05-27 04:30:03 +00:00
|
|
|
|
2009-11-17 00:12:54 +00:00
|
|
|
stream.write_function(&stream, " and hostname='%s'", switch_core_get_variable("hostname"));
|
|
|
|
switch_cache_db_execute_sql_callback(db, stream.data, comp_callback, &h, &errmsg);
|
2009-12-11 05:14:23 +00:00
|
|
|
|
2008-04-22 20:49:39 +00:00
|
|
|
if (errmsg) {
|
2008-04-23 16:46:24 +00:00
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_ERROR, "error [%s][%s]\n", (char *) stream.data, errmsg);
|
2008-04-22 20:49:39 +00:00
|
|
|
free(errmsg);
|
|
|
|
ret = CC_ERROR;
|
2008-04-23 16:46:24 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
switch_safe_free(dupdup);
|
|
|
|
switch_safe_free(stream.data);
|
2008-05-27 04:30:03 +00:00
|
|
|
|
2008-04-23 16:46:24 +00:00
|
|
|
if (ret == CC_ERROR) {
|
2008-04-22 20:49:39 +00:00
|
|
|
goto end;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-12-15 00:34:02 +00:00
|
|
|
if (h.out) {
|
|
|
|
fprintf(h.out, "\n\n");
|
|
|
|
}
|
|
|
|
|
|
|
|
if (h.stream) {
|
|
|
|
h.stream->write_function(h.stream, "\n\n");
|
|
|
|
if (h.hits == 1 && !zstr(h.last)) {
|
|
|
|
h.stream->write_function(h.stream, "write=%d:%s ", h.len, h.last);
|
|
|
|
} else if (h.hits > 1 && !zstr(h.partial)) {
|
|
|
|
h.stream->write_function(h.stream, "write=%d:%s", h.len, h.partial);
|
|
|
|
}
|
|
|
|
}
|
2008-05-27 04:30:03 +00:00
|
|
|
|
2009-12-15 00:34:02 +00:00
|
|
|
if (h.out) {
|
|
|
|
if (h.hits == 1 && !zstr(h.last)) {
|
|
|
|
el_deletestr(el, h.len);
|
|
|
|
el_insertstr(el, h.last);
|
|
|
|
el_insertstr(el, " ");
|
|
|
|
} else if (h.hits > 1 && !zstr(h.partial)) {
|
|
|
|
el_deletestr(el, h.len);
|
|
|
|
el_insertstr(el, h.partial);
|
|
|
|
}
|
2008-04-22 20:49:39 +00:00
|
|
|
}
|
|
|
|
|
2008-05-27 04:30:03 +00:00
|
|
|
end:
|
2008-04-22 20:49:39 +00:00
|
|
|
|
2009-12-15 00:34:02 +00:00
|
|
|
if (h.out) {
|
|
|
|
fflush(h.out);
|
|
|
|
}
|
2008-04-22 20:49:39 +00:00
|
|
|
|
|
|
|
switch_safe_free(sql);
|
|
|
|
switch_safe_free(dup);
|
|
|
|
|
2009-11-17 00:12:54 +00:00
|
|
|
switch_cache_db_release_db_handle(&db);
|
2008-05-27 04:30:03 +00:00
|
|
|
|
2008-04-22 20:49:39 +00:00
|
|
|
return (ret);
|
|
|
|
}
|
|
|
|
|
2009-12-15 00:34:02 +00:00
|
|
|
static unsigned char complete(EditLine * el, int ch)
|
|
|
|
{
|
|
|
|
const LineInfo *lf = el_line(el);
|
|
|
|
|
|
|
|
return switch_console_complete(lf->buffer, lf->cursor, switch_core_get_console(), NULL);
|
|
|
|
}
|
|
|
|
|
|
|
|
static struct {
|
|
|
|
switch_hash_t *func_hash;
|
|
|
|
switch_mutex_t *func_mutex;
|
|
|
|
} globals;
|
|
|
|
|
|
|
|
SWITCH_DECLARE(switch_status_t) switch_console_init(switch_memory_pool_t *pool)
|
|
|
|
{
|
|
|
|
switch_mutex_init(&globals.func_mutex, SWITCH_MUTEX_NESTED, pool);
|
|
|
|
switch_core_hash_init(&globals.func_hash, pool);
|
|
|
|
switch_console_add_complete_func("::console::list_uuid", switch_console_list_uuid);
|
|
|
|
return SWITCH_STATUS_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
SWITCH_DECLARE(switch_status_t) switch_console_shutdown(void)
|
|
|
|
{
|
|
|
|
return switch_core_hash_destroy(&globals.func_hash);
|
|
|
|
}
|
|
|
|
|
|
|
|
SWITCH_DECLARE(switch_status_t) switch_console_add_complete_func(const char *name, switch_console_complete_callback_t cb)
|
|
|
|
{
|
|
|
|
switch_status_t status;
|
|
|
|
|
|
|
|
switch_mutex_lock(globals.func_mutex);
|
|
|
|
status = switch_core_hash_insert(globals.func_hash, name, (void *)(intptr_t)cb);
|
|
|
|
switch_mutex_unlock(globals.func_mutex);
|
|
|
|
|
|
|
|
return status;
|
|
|
|
}
|
|
|
|
|
|
|
|
SWITCH_DECLARE(switch_status_t) switch_console_del_complete_func(const char *name)
|
|
|
|
{
|
|
|
|
switch_status_t status;
|
|
|
|
|
|
|
|
switch_mutex_lock(globals.func_mutex);
|
|
|
|
status = switch_core_hash_insert(globals.func_hash, name, NULL);
|
|
|
|
switch_mutex_unlock(globals.func_mutex);
|
|
|
|
|
|
|
|
return status;
|
|
|
|
}
|
|
|
|
|
|
|
|
SWITCH_DECLARE(void) switch_console_free_matches(switch_console_callback_match_t **matches)
|
|
|
|
{
|
|
|
|
switch_console_callback_match_t *my_match = *matches;
|
|
|
|
switch_console_callback_match_node_t *m, *cur;
|
|
|
|
|
|
|
|
/* Don't play with matches */
|
|
|
|
*matches = NULL;
|
|
|
|
|
|
|
|
m = my_match->head;
|
|
|
|
while(m) {
|
|
|
|
cur = m;
|
|
|
|
m = m->next;
|
|
|
|
free(cur->val);
|
|
|
|
free(cur);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
SWITCH_DECLARE(void) switch_console_push_match(switch_console_callback_match_t **matches, const char *new_val)
|
|
|
|
{
|
|
|
|
switch_console_callback_match_node_t *match;
|
|
|
|
|
|
|
|
if (!*matches) {
|
|
|
|
switch_zmalloc(*matches, sizeof(**matches));
|
|
|
|
}
|
|
|
|
|
|
|
|
switch_zmalloc(match, sizeof(*match));
|
|
|
|
match->val = strdup(new_val);
|
|
|
|
|
|
|
|
if ((*matches)->head) {
|
|
|
|
(*matches)->end->next = match;
|
|
|
|
} else {
|
|
|
|
(*matches)->head = match;
|
|
|
|
}
|
|
|
|
|
|
|
|
(*matches)->end = match;
|
|
|
|
}
|
|
|
|
|
|
|
|
SWITCH_DECLARE(switch_status_t) switch_console_run_complete_func(const char *func, const char *line, const char *last_word,
|
|
|
|
switch_console_callback_match_t **matches)
|
|
|
|
{
|
|
|
|
switch_console_complete_callback_t cb;
|
|
|
|
switch_status_t status = SWITCH_STATUS_FALSE;
|
|
|
|
|
|
|
|
switch_mutex_lock(globals.func_mutex);
|
|
|
|
if ((cb = (switch_console_complete_callback_t)(intptr_t)switch_core_hash_find(globals.func_hash, func))) {
|
|
|
|
status = cb(line, last_word, matches);
|
|
|
|
}
|
|
|
|
switch_mutex_unlock(globals.func_mutex);
|
|
|
|
|
|
|
|
return status;
|
|
|
|
}
|
|
|
|
|
2008-04-23 16:46:24 +00:00
|
|
|
|
|
|
|
SWITCH_DECLARE(switch_status_t) switch_console_set_complete(const char *string)
|
|
|
|
{
|
2008-05-27 04:30:03 +00:00
|
|
|
char *mydata = NULL, *argv[11] = { 0 };
|
2008-04-23 16:46:24 +00:00
|
|
|
int argc, x;
|
|
|
|
switch_status_t status = SWITCH_STATUS_FALSE;
|
|
|
|
|
|
|
|
if (string && (mydata = strdup(string))) {
|
|
|
|
if ((argc = switch_separate_string(mydata, ' ', argv, (sizeof(argv) / sizeof(argv[0]))))) {
|
2009-11-17 00:12:54 +00:00
|
|
|
switch_cache_db_handle_t *db = NULL;
|
2008-04-23 16:46:24 +00:00
|
|
|
switch_stream_handle_t mystream = { 0 };
|
|
|
|
SWITCH_STANDARD_STREAM(mystream);
|
|
|
|
|
2009-11-17 00:12:54 +00:00
|
|
|
switch_core_db_handle(&db);
|
2009-12-10 20:51:13 +00:00
|
|
|
|
2008-04-23 22:23:23 +00:00
|
|
|
if (!strcasecmp(argv[0], "stickyadd")) {
|
|
|
|
mystream.write_function(&mystream, "insert into complete values (1,");
|
2008-05-27 04:30:03 +00:00
|
|
|
for (x = 0; x < 10; x++) {
|
2009-12-10 20:51:13 +00:00
|
|
|
if (argv[x + 1] && !strcasecmp(argv[x + 1], "_any_")) {
|
|
|
|
mystream.write_function(&mystream, "%s", "'', ");
|
|
|
|
} else {
|
|
|
|
if (db->type == SCDB_TYPE_CORE_DB) {
|
|
|
|
mystream.write_function(&mystream, "'%q', ", switch_str_nil(argv[x + 1]));
|
|
|
|
} else {
|
|
|
|
mystream.write_function(&mystream, "'%w', ", switch_str_nil(argv[x + 1]));
|
|
|
|
}
|
|
|
|
}
|
2008-04-23 22:23:23 +00:00
|
|
|
}
|
2009-11-17 00:12:54 +00:00
|
|
|
mystream.write_function(&mystream, " '%s')", switch_core_get_variable("hostname"));
|
|
|
|
switch_cache_db_persistant_execute(db, mystream.data, 5);
|
2008-04-23 22:23:23 +00:00
|
|
|
status = SWITCH_STATUS_SUCCESS;
|
|
|
|
} else if (!strcasecmp(argv[0], "add")) {
|
|
|
|
mystream.write_function(&mystream, "insert into complete values (0,");
|
2008-05-27 04:30:03 +00:00
|
|
|
for (x = 0; x < 10; x++) {
|
2009-12-10 20:51:13 +00:00
|
|
|
if (argv[x + 1] && !strcasecmp(argv[x + 1], "_any_")) {
|
|
|
|
mystream.write_function(&mystream, "%s", "'', ");
|
|
|
|
} else {
|
|
|
|
if (db->type == SCDB_TYPE_CORE_DB) {
|
|
|
|
mystream.write_function(&mystream, "'%q', ", switch_str_nil(argv[x + 1]));
|
|
|
|
} else {
|
|
|
|
mystream.write_function(&mystream, "'%w', ", switch_str_nil(argv[x + 1]));
|
|
|
|
}
|
|
|
|
}
|
2008-04-23 16:46:24 +00:00
|
|
|
}
|
2009-11-17 00:12:54 +00:00
|
|
|
mystream.write_function(&mystream, " '%s')", switch_core_get_variable("hostname"));
|
2009-12-10 20:51:13 +00:00
|
|
|
|
2009-11-17 00:12:54 +00:00
|
|
|
switch_cache_db_persistant_execute(db, mystream.data, 5);
|
2008-04-23 16:46:24 +00:00
|
|
|
status = SWITCH_STATUS_SUCCESS;
|
|
|
|
} else if (!strcasecmp(argv[0], "del")) {
|
|
|
|
char *what = argv[1];
|
2008-05-27 04:30:03 +00:00
|
|
|
if (!strcasecmp(what, "*")) {
|
2009-11-17 00:12:54 +00:00
|
|
|
switch_cache_db_persistant_execute(db, "delete from complete", 1);
|
2008-04-23 16:46:24 +00:00
|
|
|
} else {
|
|
|
|
mystream.write_function(&mystream, "delete from complete where ");
|
2008-05-27 04:30:03 +00:00
|
|
|
for (x = 0; x < argc - 1; x++) {
|
2009-12-10 20:51:13 +00:00
|
|
|
if (db->type == SCDB_TYPE_CORE_DB) {
|
|
|
|
mystream.write_function(&mystream, "a%d = '%q'%q", x + 1, switch_str_nil(argv[x + 1]), x == argc - 2 ? "" : " and ");
|
|
|
|
} else {
|
|
|
|
mystream.write_function(&mystream, "a%d = '%w'%w", x + 1, switch_str_nil(argv[x + 1]), x == argc - 2 ? "" : " and ");
|
|
|
|
}
|
2008-04-23 16:46:24 +00:00
|
|
|
}
|
2009-11-17 00:12:54 +00:00
|
|
|
mystream.write_function(&mystream, " and hostname='%s'", switch_core_get_variable("hostname"));
|
|
|
|
switch_cache_db_persistant_execute(db, mystream.data, 1);
|
2008-04-23 16:46:24 +00:00
|
|
|
}
|
|
|
|
status = SWITCH_STATUS_SUCCESS;
|
|
|
|
}
|
2009-12-10 20:51:13 +00:00
|
|
|
|
2008-04-23 16:46:24 +00:00
|
|
|
switch_safe_free(mystream.data);
|
2009-11-17 00:12:54 +00:00
|
|
|
switch_cache_db_release_db_handle(&db);
|
2008-04-23 16:46:24 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
switch_safe_free(mydata);
|
|
|
|
|
|
|
|
return status;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-04-23 17:38:13 +00:00
|
|
|
SWITCH_DECLARE(switch_status_t) switch_console_set_alias(const char *string)
|
|
|
|
{
|
2008-05-27 04:30:03 +00:00
|
|
|
char *mydata = NULL, *argv[3] = { 0 };
|
2008-04-23 17:38:13 +00:00
|
|
|
int argc;
|
|
|
|
switch_status_t status = SWITCH_STATUS_FALSE;
|
2008-05-27 04:30:03 +00:00
|
|
|
|
2008-04-23 17:38:13 +00:00
|
|
|
if (string && (mydata = strdup(string))) {
|
|
|
|
if ((argc = switch_separate_string(mydata, ' ', argv, (sizeof(argv) / sizeof(argv[0])))) >= 2) {
|
2009-11-17 00:12:54 +00:00
|
|
|
switch_cache_db_handle_t *db = NULL;
|
2008-04-23 17:38:13 +00:00
|
|
|
char *sql = NULL;
|
|
|
|
|
2009-11-17 00:12:54 +00:00
|
|
|
switch_core_db_handle(&db);
|
2008-04-23 22:23:23 +00:00
|
|
|
|
|
|
|
if (!strcasecmp(argv[0], "stickyadd") && argc == 3) {
|
2009-11-17 00:12:54 +00:00
|
|
|
sql = switch_mprintf("delete from aliases where alias='%q' and hostname='%q'", argv[1], switch_core_get_variable("hostname"));
|
|
|
|
switch_cache_db_persistant_execute(db, sql, 5);
|
2008-04-23 22:23:23 +00:00
|
|
|
switch_safe_free(sql);
|
2009-12-10 20:51:13 +00:00
|
|
|
if (db->type == SCDB_TYPE_CORE_DB) {
|
|
|
|
sql = switch_mprintf("insert into aliases (sticky, alias, command, hostname) values (1, '%q','%q','%q')",
|
|
|
|
argv[1], argv[2], switch_core_get_variable("hostname"));
|
|
|
|
} else {
|
|
|
|
sql = switch_mprintf("insert into aliases (sticky, alias, command, hostname) values (1, '%w','%w','%w')",
|
|
|
|
argv[1], argv[2], switch_core_get_variable("hostname"));
|
|
|
|
}
|
2009-11-17 00:12:54 +00:00
|
|
|
switch_cache_db_persistant_execute(db, sql, 5);
|
2008-04-23 22:23:23 +00:00
|
|
|
status = SWITCH_STATUS_SUCCESS;
|
|
|
|
} else if (!strcasecmp(argv[0], "add") && argc == 3) {
|
2009-11-17 00:12:54 +00:00
|
|
|
sql = switch_mprintf("delete from aliases where alias='%q' and hostname='%q'", argv[1], switch_core_get_variable("hostname"));
|
|
|
|
switch_cache_db_persistant_execute(db, sql, 5);
|
2008-04-23 17:38:13 +00:00
|
|
|
switch_safe_free(sql);
|
2009-12-10 20:51:13 +00:00
|
|
|
if (db->type == SCDB_TYPE_CORE_DB) {
|
|
|
|
sql = switch_mprintf("insert into aliases (sticky, alias, command, hostname) values (0, '%q','%q','%q')",
|
|
|
|
argv[1], argv[2], switch_core_get_variable("hostname"));
|
|
|
|
} else {
|
|
|
|
sql = switch_mprintf("insert into aliases (sticky, alias, command, hostname) values (0, '%w','%w','%w')",
|
|
|
|
argv[1], argv[2], switch_core_get_variable("hostname"));
|
|
|
|
}
|
2009-11-17 00:12:54 +00:00
|
|
|
switch_cache_db_persistant_execute(db, sql, 5);
|
2008-04-23 17:38:13 +00:00
|
|
|
status = SWITCH_STATUS_SUCCESS;
|
|
|
|
} else if (!strcasecmp(argv[0], "del") && argc == 2) {
|
|
|
|
char *what = argv[1];
|
2008-05-27 04:30:03 +00:00
|
|
|
if (!strcasecmp(what, "*")) {
|
2009-11-17 00:12:54 +00:00
|
|
|
sql = switch_mprintf("delete from aliases where hostname='%q'", switch_core_get_variable("hostname"));
|
|
|
|
switch_cache_db_persistant_execute(db, sql, 1);
|
2008-04-23 17:38:13 +00:00
|
|
|
} else {
|
2009-11-17 00:12:54 +00:00
|
|
|
sql = switch_mprintf("delete from aliases where alias='%q' and hostname='%q'", argv[1], switch_core_get_variable("hostname"));
|
|
|
|
switch_cache_db_persistant_execute(db, sql, 5);
|
2008-04-23 17:38:13 +00:00
|
|
|
}
|
|
|
|
status = SWITCH_STATUS_SUCCESS;
|
|
|
|
}
|
|
|
|
switch_safe_free(sql);
|
2009-11-17 00:12:54 +00:00
|
|
|
switch_cache_db_release_db_handle(&db);
|
2008-04-23 17:38:13 +00:00
|
|
|
}
|
|
|
|
}
|
2008-05-27 04:30:03 +00:00
|
|
|
|
2008-04-23 17:38:13 +00:00
|
|
|
switch_safe_free(mydata);
|
|
|
|
|
|
|
|
return status;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2007-05-14 16:20:57 +00:00
|
|
|
SWITCH_DECLARE(void) switch_console_loop(void)
|
|
|
|
{
|
|
|
|
switch_thread_t *thread;
|
|
|
|
switch_threadattr_t *thd_attr = NULL;
|
|
|
|
switch_memory_pool_t *pool;
|
2008-03-03 19:02:36 +00:00
|
|
|
|
2009-09-01 17:48:11 +00:00
|
|
|
gethostname(hostname, sizeof(hostname));
|
|
|
|
|
2007-05-14 16:20:57 +00:00
|
|
|
if (switch_core_new_memory_pool(&pool) != SWITCH_STATUS_SUCCESS) {
|
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_CRIT, "Pool Failure\n");
|
|
|
|
return;
|
|
|
|
}
|
2008-03-03 19:02:36 +00:00
|
|
|
|
2007-05-14 16:20:57 +00:00
|
|
|
el = el_init(__FILE__, switch_core_get_console(), switch_core_get_console(), switch_core_get_console());
|
|
|
|
el_set(el, EL_PROMPT, &prompt);
|
|
|
|
el_set(el, EL_EDITOR, "emacs");
|
2008-05-27 04:30:03 +00:00
|
|
|
/* AGX: Bind Keyboard function keys. This has been tested with:
|
|
|
|
* - linux console keyabord
|
|
|
|
* - putty.exe connected via ssh to linux
|
|
|
|
*/
|
|
|
|
/* Load/Init the config first */
|
|
|
|
console_xml_config();
|
|
|
|
/* Bind the functions to the key */
|
|
|
|
el_set(el, EL_ADDFN, "f1-key", "F1 KEY PRESS", console_f1key);
|
|
|
|
el_set(el, EL_ADDFN, "f2-key", "F2 KEY PRESS", console_f2key);
|
|
|
|
el_set(el, EL_ADDFN, "f3-key", "F3 KEY PRESS", console_f3key);
|
|
|
|
el_set(el, EL_ADDFN, "f4-key", "F4 KEY PRESS", console_f4key);
|
|
|
|
el_set(el, EL_ADDFN, "f5-key", "F5 KEY PRESS", console_f5key);
|
|
|
|
el_set(el, EL_ADDFN, "f6-key", "F6 KEY PRESS", console_f6key);
|
|
|
|
el_set(el, EL_ADDFN, "f7-key", "F7 KEY PRESS", console_f7key);
|
|
|
|
el_set(el, EL_ADDFN, "f8-key", "F8 KEY PRESS", console_f8key);
|
|
|
|
el_set(el, EL_ADDFN, "f9-key", "F9 KEY PRESS", console_f9key);
|
|
|
|
el_set(el, EL_ADDFN, "f10-key", "F10 KEY PRESS", console_f10key);
|
|
|
|
el_set(el, EL_ADDFN, "f11-key", "F11 KEY PRESS", console_f11key);
|
|
|
|
el_set(el, EL_ADDFN, "f12-key", "F12 KEY PRESS", console_f12key);
|
|
|
|
|
|
|
|
el_set(el, EL_BIND, "\033OP", "f1-key", NULL);
|
|
|
|
el_set(el, EL_BIND, "\033OQ", "f2-key", NULL);
|
|
|
|
el_set(el, EL_BIND, "\033OR", "f3-key", NULL);
|
|
|
|
el_set(el, EL_BIND, "\033OS", "f4-key", NULL);
|
|
|
|
|
|
|
|
|
|
|
|
el_set(el, EL_BIND, "\033[11~", "f1-key", NULL);
|
|
|
|
el_set(el, EL_BIND, "\033[12~", "f2-key", NULL);
|
|
|
|
el_set(el, EL_BIND, "\033[13~", "f3-key", NULL);
|
|
|
|
el_set(el, EL_BIND, "\033[14~", "f4-key", NULL);
|
|
|
|
el_set(el, EL_BIND, "\033[15~", "f5-key", NULL);
|
|
|
|
el_set(el, EL_BIND, "\033[17~", "f6-key", NULL);
|
|
|
|
el_set(el, EL_BIND, "\033[18~", "f7-key", NULL);
|
|
|
|
el_set(el, EL_BIND, "\033[19~", "f8-key", NULL);
|
|
|
|
el_set(el, EL_BIND, "\033[20~", "f9-key", NULL);
|
|
|
|
el_set(el, EL_BIND, "\033[21~", "f10-key", NULL);
|
|
|
|
el_set(el, EL_BIND, "\033[23~", "f11-key", NULL);
|
|
|
|
el_set(el, EL_BIND, "\033[24~", "f12-key", NULL);
|
2007-05-14 16:20:57 +00:00
|
|
|
|
2008-04-22 20:49:39 +00:00
|
|
|
|
|
|
|
el_set(el, EL_ADDFN, "ed-complete", "Complete argument", complete);
|
|
|
|
el_set(el, EL_BIND, "^I", "ed-complete", NULL);
|
|
|
|
|
2007-05-14 16:20:57 +00:00
|
|
|
myhistory = history_init();
|
|
|
|
if (myhistory == 0) {
|
|
|
|
fprintf(stderr, "history could not be initialized\n");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
hfile = switch_mprintf("%s%sfreeswitch.history", SWITCH_GLOBAL_dirs.log_dir, SWITCH_PATH_SEPARATOR);
|
|
|
|
assert(hfile != NULL);
|
|
|
|
|
|
|
|
history(myhistory, &ev, H_SETSIZE, 800);
|
|
|
|
el_set(el, EL_HIST, history, myhistory);
|
|
|
|
history(myhistory, &ev, H_LOAD, hfile);
|
|
|
|
|
2009-06-25 19:29:34 +00:00
|
|
|
el_source(el, NULL);
|
|
|
|
|
2007-05-14 16:20:57 +00:00
|
|
|
switch_threadattr_create(&thd_attr, pool);
|
|
|
|
switch_threadattr_detach_set(thd_attr, 1);
|
|
|
|
switch_threadattr_stacksize_set(thd_attr, SWITCH_THREAD_STACKSIZE);
|
|
|
|
switch_thread_create(&thread, thd_attr, console_thread, pool, pool);
|
2008-05-27 04:30:03 +00:00
|
|
|
|
2007-05-14 16:20:57 +00:00
|
|
|
while (running) {
|
2007-10-03 23:43:01 +00:00
|
|
|
int32_t arg = 0;
|
2007-05-14 16:20:57 +00:00
|
|
|
switch_core_session_ctl(SCSC_CHECK_RUNNING, &arg);
|
|
|
|
if (!arg) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
switch_yield(1000000);
|
|
|
|
}
|
|
|
|
|
|
|
|
history(myhistory, &ev, H_SAVE, hfile);
|
2007-04-30 20:37:41 +00:00
|
|
|
free(hfile);
|
2008-05-27 04:30:03 +00:00
|
|
|
|
2007-04-30 20:37:41 +00:00
|
|
|
/* Clean up our memory */
|
|
|
|
history_end(myhistory);
|
|
|
|
el_end(el);
|
|
|
|
}
|
|
|
|
|
|
|
|
#else
|
2008-04-26 17:02:43 +00:00
|
|
|
SWITCH_DECLARE(switch_status_t) switch_console_set_alias(const char *string)
|
|
|
|
{
|
|
|
|
return SWITCH_STATUS_FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
SWITCH_DECLARE(switch_status_t) switch_console_set_complete(const char *string)
|
|
|
|
{
|
|
|
|
return SWITCH_STATUS_FALSE;
|
|
|
|
}
|
2007-04-30 20:37:41 +00:00
|
|
|
|
|
|
|
SWITCH_DECLARE(void) switch_console_loop(void)
|
|
|
|
{
|
|
|
|
|
2009-10-16 15:48:40 +00:00
|
|
|
char cmd[2048] = "";
|
2009-10-23 15:15:34 +00:00
|
|
|
int32_t activity = 1;
|
2005-11-19 20:07:43 +00:00
|
|
|
gethostname(hostname, sizeof(hostname));
|
2009-12-15 00:34:02 +00:00
|
|
|
|
2007-03-29 22:31:56 +00:00
|
|
|
while (running) {
|
2007-10-04 13:17:56 +00:00
|
|
|
int32_t arg;
|
2009-10-16 15:48:40 +00:00
|
|
|
#ifdef _MSC_VER
|
2009-10-23 15:15:34 +00:00
|
|
|
DWORD read, i;
|
2009-10-16 15:48:40 +00:00
|
|
|
HANDLE stdinHandle = GetStdHandle(STD_INPUT_HANDLE);
|
|
|
|
INPUT_RECORD in[128];
|
|
|
|
#else
|
2006-09-20 20:25:26 +00:00
|
|
|
fd_set rfds, efds;
|
2007-03-29 22:31:56 +00:00
|
|
|
struct timeval tv = { 0, 20000 };
|
2007-02-14 00:31:11 +00:00
|
|
|
#endif
|
2006-09-20 20:25:26 +00:00
|
|
|
|
|
|
|
switch_core_session_ctl(SCSC_CHECK_RUNNING, &arg);
|
|
|
|
if (!arg) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2005-11-19 20:07:43 +00:00
|
|
|
if (activity) {
|
2006-04-11 21:13:44 +00:00
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG_CLEAN, SWITCH_LOG_CONSOLE, "\nfreeswitch@%s> ", hostname);
|
2005-11-19 20:07:43 +00:00
|
|
|
}
|
2006-09-20 20:47:28 +00:00
|
|
|
#ifdef _MSC_VER
|
2009-10-16 15:48:40 +00:00
|
|
|
activity = 0;
|
2009-10-23 15:15:34 +00:00
|
|
|
PeekConsoleInput(stdinHandle, in, 128, &read);
|
|
|
|
for (i = 0; i < read; i++) {
|
|
|
|
if (in[i].EventType == KEY_EVENT && !in[i].Event.KeyEvent.bKeyDown) {
|
|
|
|
activity = 1;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2009-10-16 15:48:40 +00:00
|
|
|
|
2009-10-23 15:15:34 +00:00
|
|
|
if (activity) {
|
2009-10-16 15:48:40 +00:00
|
|
|
DWORD bytes = 0;
|
2009-10-23 15:15:34 +00:00
|
|
|
char *end;
|
|
|
|
ReadConsole(stdinHandle, cmd, sizeof(cmd), &bytes, NULL);
|
|
|
|
FlushConsoleInputBuffer(stdinHandle);
|
|
|
|
end = end_of_p(cmd);
|
|
|
|
while(*end == '\r' || *end == '\n') {
|
|
|
|
*end-- = '\0';
|
2009-10-16 15:48:40 +00:00
|
|
|
}
|
2009-10-23 15:15:34 +00:00
|
|
|
}
|
2006-09-20 20:47:28 +00:00
|
|
|
|
2009-10-23 15:15:34 +00:00
|
|
|
if (cmd[0]) {
|
|
|
|
running = switch_console_process(cmd, 0);
|
|
|
|
memset(cmd, 0, sizeof(cmd));
|
2007-02-14 00:31:11 +00:00
|
|
|
}
|
2009-10-23 15:15:34 +00:00
|
|
|
Sleep(20);
|
2007-02-14 00:31:11 +00:00
|
|
|
#else
|
2006-09-20 20:25:26 +00:00
|
|
|
FD_ZERO(&rfds);
|
|
|
|
FD_ZERO(&efds);
|
|
|
|
FD_SET(fileno(stdin), &rfds);
|
|
|
|
FD_SET(fileno(stdin), &efds);
|
2007-03-29 22:31:56 +00:00
|
|
|
if ((activity = select(fileno(stdin) + 1, &rfds, NULL, &efds, &tv)) < 0) {
|
|
|
|
break;
|
|
|
|
}
|
2005-11-19 20:07:43 +00:00
|
|
|
|
2009-09-10 23:00:16 +00:00
|
|
|
if (FD_ISSET(fileno(stdin), &efds)) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!FD_ISSET(fileno(stdin), &rfds)) {
|
|
|
|
activity = 0;
|
|
|
|
}
|
|
|
|
|
2005-11-19 20:07:43 +00:00
|
|
|
if (activity == 0) {
|
|
|
|
fflush(stdout);
|
|
|
|
continue;
|
|
|
|
}
|
2009-10-16 15:48:40 +00:00
|
|
|
|
2005-11-19 20:07:43 +00:00
|
|
|
memset(&cmd, 0, sizeof(cmd));
|
2007-03-29 22:31:56 +00:00
|
|
|
for (x = 0; x < (sizeof(cmd) - 1); x++) {
|
|
|
|
int c = getchar();
|
|
|
|
if (c < 0) {
|
2008-05-15 21:25:26 +00:00
|
|
|
int y = read(fileno(stdin), cmd, sizeof(cmd) - 1);
|
2007-03-29 22:31:56 +00:00
|
|
|
cmd[y - 1] = '\0';
|
|
|
|
break;
|
|
|
|
}
|
2006-12-19 19:15:13 +00:00
|
|
|
|
|
|
|
cmd[x] = (char) c;
|
|
|
|
|
2005-11-19 20:07:43 +00:00
|
|
|
if (cmd[x] == '\n') {
|
|
|
|
cmd[x] = '\0';
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2007-03-29 22:31:56 +00:00
|
|
|
|
2006-01-20 15:05:05 +00:00
|
|
|
if (cmd[0]) {
|
2008-04-23 17:38:13 +00:00
|
|
|
running = switch_console_process(cmd, 0);
|
2005-11-19 20:07:43 +00:00
|
|
|
}
|
2009-10-16 15:48:40 +00:00
|
|
|
#endif
|
2005-11-19 20:07:43 +00:00
|
|
|
}
|
|
|
|
}
|
2007-04-30 20:37:41 +00:00
|
|
|
#endif
|
2006-11-27 22:30:48 +00:00
|
|
|
|
|
|
|
/* For Emacs:
|
|
|
|
* Local Variables:
|
|
|
|
* mode:c
|
2008-02-03 22:14:57 +00:00
|
|
|
* indent-tabs-mode:t
|
2006-11-27 22:30:48 +00:00
|
|
|
* tab-width:4
|
|
|
|
* c-basic-offset:4
|
|
|
|
* End:
|
|
|
|
* For VIM:
|
2008-07-03 19:12:26 +00:00
|
|
|
* vim:set softtabstop=4 shiftwidth=4 tabstop=4:
|
2006-11-27 22:30:48 +00:00
|
|
|
*/
|