2007-10-01 20:39:16 +00:00
|
|
|
/*
|
|
|
|
* FreeSWITCH Modular Media Switching Software Library / Soft-Switch Application
|
|
|
|
* Copyright (C) 2005/2006, Anthony Minessale II <anthmct@yahoo.com>
|
|
|
|
*
|
|
|
|
* Version: MPL 1.1
|
|
|
|
*
|
|
|
|
* The contents of this file are subject to the Mozilla Public License Version
|
|
|
|
* 1.1 (the "License"); you may not use this file except in compliance with
|
|
|
|
* the License. You may obtain a copy of the License at
|
|
|
|
* http://www.mozilla.org/MPL/
|
|
|
|
*
|
|
|
|
* Software distributed under the License is distributed on an "AS IS" basis,
|
|
|
|
* WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License
|
|
|
|
* for the specific language governing rights and limitations under the
|
|
|
|
* License.
|
|
|
|
*
|
|
|
|
* The Original Code is FreeSWITCH Modular Media Switching Software Library / Soft-Switch Application
|
|
|
|
*
|
|
|
|
* The Initial Developer of the Original Code is
|
|
|
|
* Anthony Minessale II <anthmct@yahoo.com>
|
|
|
|
* Portions created by the Initial Developer are Copyright (C)
|
|
|
|
* the Initial Developer. All Rights Reserved.
|
|
|
|
*
|
|
|
|
* Contributor(s):
|
|
|
|
*
|
|
|
|
* Anthony Minessale II <anthmct@yahoo.com>
|
|
|
|
*
|
|
|
|
* mod_fifo.c -- FIFO
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
#include <switch.h>
|
|
|
|
|
|
|
|
SWITCH_MODULE_SHUTDOWN_FUNCTION(mod_fifo_shutdown);
|
|
|
|
SWITCH_MODULE_LOAD_FUNCTION(mod_fifo_load);
|
|
|
|
SWITCH_MODULE_DEFINITION(mod_fifo, mod_fifo_load, mod_fifo_shutdown, NULL);
|
|
|
|
|
2007-10-02 19:58:06 +00:00
|
|
|
#define FIFO_EVENT "fifo::info"
|
|
|
|
|
2008-03-19 16:03:51 +00:00
|
|
|
#define MAX_PRI 10
|
|
|
|
|
|
|
|
struct fifo_node {
|
|
|
|
char *name;
|
|
|
|
switch_mutex_t *mutex;
|
|
|
|
switch_queue_t *fifo_list[MAX_PRI];
|
|
|
|
switch_hash_t *caller_hash;
|
|
|
|
switch_hash_t *consumer_hash;
|
|
|
|
int caller_count;
|
|
|
|
int consumer_count;
|
2008-04-04 21:18:16 +00:00
|
|
|
switch_time_t start_waiting;
|
|
|
|
uint32_t importance;
|
2008-03-19 16:03:51 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
typedef struct fifo_node fifo_node_t;
|
2007-10-02 00:03:53 +00:00
|
|
|
|
|
|
|
|
|
|
|
static switch_status_t on_dtmf(switch_core_session_t *session, void *input, switch_input_type_t itype, void *buf, unsigned int buflen)
|
|
|
|
{
|
2008-01-28 07:26:10 +00:00
|
|
|
switch_core_session_t *bleg = (switch_core_session_t *) buf;
|
2007-10-02 00:03:53 +00:00
|
|
|
|
|
|
|
switch (itype) {
|
|
|
|
case SWITCH_INPUT_TYPE_DTMF:
|
|
|
|
{
|
2007-12-22 00:32:20 +00:00
|
|
|
switch_dtmf_t *dtmf = (switch_dtmf_t *) input;
|
|
|
|
|
|
|
|
if (dtmf->digit == '*') {
|
2008-01-28 07:26:10 +00:00
|
|
|
if (switch_channel_test_flag(switch_core_session_get_channel(session), CF_ORIGINATOR)) {
|
|
|
|
switch_channel_hangup(switch_core_session_get_channel(bleg), SWITCH_CAUSE_NORMAL_CLEARING);
|
2007-10-02 00:03:53 +00:00
|
|
|
return SWITCH_STATUS_BREAK;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
return SWITCH_STATUS_SUCCESS;
|
|
|
|
}
|
|
|
|
|
2008-04-03 21:27:16 +00:00
|
|
|
|
|
|
|
static switch_status_t moh_on_dtmf(switch_core_session_t *session, void *input, switch_input_type_t itype, void *buf, unsigned int buflen)
|
|
|
|
{
|
|
|
|
|
|
|
|
switch (itype) {
|
|
|
|
case SWITCH_INPUT_TYPE_DTMF:
|
|
|
|
{
|
|
|
|
switch_dtmf_t *dtmf = (switch_dtmf_t *) input;
|
2008-04-03 21:31:04 +00:00
|
|
|
switch_channel_t *channel = switch_core_session_get_channel(session);
|
|
|
|
const char *caller_exit_key = switch_channel_get_variable(channel, "fifo_caller_exit_key");
|
|
|
|
|
|
|
|
if (caller_exit_key && dtmf->digit == *caller_exit_key) {
|
2008-04-03 21:27:16 +00:00
|
|
|
char *bp = buf;
|
|
|
|
*bp = dtmf->digit;
|
|
|
|
return SWITCH_STATUS_BREAK;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
return SWITCH_STATUS_SUCCESS;
|
|
|
|
}
|
|
|
|
|
2007-10-02 00:03:53 +00:00
|
|
|
#define check_string(s) if (!switch_strlen_zero(s) && !strcasecmp(s, "undef")) { s = NULL; }
|
|
|
|
|
2008-04-04 21:18:16 +00:00
|
|
|
static int node_consumer_wait_count(fifo_node_t *node)
|
2007-10-02 00:03:53 +00:00
|
|
|
{
|
2008-04-04 21:18:16 +00:00
|
|
|
int i, len = 0;
|
2008-03-19 16:03:51 +00:00
|
|
|
|
2008-04-04 21:18:16 +00:00
|
|
|
for (i = 0; i < MAX_PRI; i++) {
|
|
|
|
len += switch_queue_size(node->fifo_list[i]);
|
|
|
|
}
|
|
|
|
|
|
|
|
return len;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void node_remove_uuid(fifo_node_t *node, const char *uuid)
|
|
|
|
{
|
|
|
|
int i, len = 0;
|
|
|
|
void *pop = NULL;
|
|
|
|
|
|
|
|
for (i = 0; i < MAX_PRI; i++) {
|
|
|
|
if (!(len = switch_queue_size(node->fifo_list[i]))) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
while(len) {
|
|
|
|
if (switch_queue_trypop(node->fifo_list[i], &pop) == SWITCH_STATUS_SUCCESS && pop) {
|
|
|
|
if (!strcmp((char *)pop, uuid)) {
|
|
|
|
free(pop);
|
|
|
|
goto end;
|
|
|
|
}
|
|
|
|
switch_queue_push(node->fifo_list[i], pop);
|
|
|
|
}
|
|
|
|
len--;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
end:
|
|
|
|
|
|
|
|
if (!node_consumer_wait_count(node)) {
|
|
|
|
node->start_waiting = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
#define MAX_CHIME 25
|
|
|
|
struct fifo_chime_data {
|
|
|
|
char *list[MAX_CHIME];
|
|
|
|
int total;
|
|
|
|
int index;
|
|
|
|
time_t next;
|
|
|
|
int freq;
|
|
|
|
int abort;
|
2008-04-04 21:48:03 +00:00
|
|
|
time_t orbit_timeout;
|
2008-04-04 21:18:16 +00:00
|
|
|
int do_orbit;
|
|
|
|
char *orbit_exten;
|
|
|
|
};
|
|
|
|
typedef struct fifo_chime_data fifo_chime_data_t;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
static switch_status_t caller_read_frame_callback(switch_core_session_t *session, switch_frame_t *frame, void *user_data)
|
|
|
|
{
|
|
|
|
fifo_chime_data_t *cd = (fifo_chime_data_t *) user_data;
|
|
|
|
|
2008-04-04 22:02:08 +00:00
|
|
|
if (!cd) {
|
|
|
|
return SWITCH_STATUS_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (cd->total && switch_timestamp(NULL) >= cd->next) {
|
2008-04-04 21:18:16 +00:00
|
|
|
if (cd->index == MAX_CHIME || cd->index == cd->total || !cd->list[cd->index]) {
|
|
|
|
cd->index = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (cd->list[cd->index]) {
|
|
|
|
switch_input_args_t args = { 0 };
|
|
|
|
char buf[25] = "";
|
|
|
|
switch_channel_t *channel = switch_core_session_get_channel(session);
|
|
|
|
const char *caller_exit_key = switch_channel_get_variable(channel, "fifo_caller_exit_key");
|
|
|
|
args.input_callback = moh_on_dtmf;
|
|
|
|
args.buf = buf;
|
|
|
|
args.buflen = sizeof(buf);
|
|
|
|
switch_ivr_play_file(session, NULL, cd->list[cd->index], &args);
|
|
|
|
if (caller_exit_key && *buf == *caller_exit_key) {
|
|
|
|
cd->abort = 1;
|
|
|
|
return SWITCH_STATUS_FALSE;
|
|
|
|
}
|
|
|
|
cd->next = switch_timestamp(NULL) + cd->freq;
|
|
|
|
cd->index++;
|
|
|
|
}
|
|
|
|
} else if (cd->orbit_timeout && switch_timestamp(NULL) >= cd->orbit_timeout) {
|
|
|
|
cd->do_orbit = 1;
|
|
|
|
return SWITCH_STATUS_FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
return SWITCH_STATUS_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static switch_status_t consumer_read_frame_callback(switch_core_session_t *session, switch_frame_t *frame, void *user_data)
|
|
|
|
{
|
|
|
|
fifo_node_t *node, **node_list = (fifo_node_t **) user_data;
|
|
|
|
int x = 0, total = 0, i = 0;
|
|
|
|
|
|
|
|
for(i = 0; ; i++) {
|
|
|
|
if (!(node = node_list[i])) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
for (x = 0; x < MAX_PRI; x++) {
|
|
|
|
total += switch_queue_size(node->fifo_list[x]);
|
|
|
|
}
|
2008-03-19 16:03:51 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (total) {
|
2007-10-02 00:03:53 +00:00
|
|
|
return SWITCH_STATUS_FALSE;
|
|
|
|
}
|
2008-03-19 16:03:51 +00:00
|
|
|
|
2007-10-02 00:03:53 +00:00
|
|
|
return SWITCH_STATUS_SUCCESS;
|
|
|
|
}
|
|
|
|
|
2007-10-01 20:39:16 +00:00
|
|
|
static struct {
|
|
|
|
switch_hash_t *fifo_hash;
|
2007-10-02 19:58:06 +00:00
|
|
|
switch_mutex_t *mutex;
|
2007-10-01 20:39:16 +00:00
|
|
|
switch_memory_pool_t *pool;
|
2008-01-03 21:34:44 +00:00
|
|
|
int running;
|
2007-10-01 20:39:16 +00:00
|
|
|
} globals;
|
|
|
|
|
2007-10-02 19:58:06 +00:00
|
|
|
|
2008-04-04 21:18:16 +00:00
|
|
|
static fifo_node_t *create_node(const char *name, uint32_t importance)
|
2007-12-18 23:07:48 +00:00
|
|
|
{
|
|
|
|
fifo_node_t *node;
|
2008-03-19 16:03:51 +00:00
|
|
|
int x = 0;
|
|
|
|
|
2008-01-03 21:34:44 +00:00
|
|
|
if (!globals.running) {
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2007-12-18 23:07:48 +00:00
|
|
|
node = switch_core_alloc(globals.pool, sizeof(*node));
|
|
|
|
node->name = switch_core_strdup(globals.pool, name);
|
2008-03-19 16:03:51 +00:00
|
|
|
for (x = 0; x < MAX_PRI; x++) {
|
|
|
|
switch_queue_create(&node->fifo_list[x], SWITCH_CORE_QUEUE_LEN, globals.pool);
|
|
|
|
switch_assert(node->fifo_list[x]);
|
|
|
|
}
|
2007-12-18 23:07:48 +00:00
|
|
|
switch_core_hash_init(&node->caller_hash, globals.pool);
|
|
|
|
switch_core_hash_init(&node->consumer_hash, globals.pool);
|
2008-03-19 16:03:51 +00:00
|
|
|
|
2007-12-18 23:07:48 +00:00
|
|
|
switch_mutex_init(&node->mutex, SWITCH_MUTEX_NESTED, globals.pool);
|
|
|
|
switch_core_hash_insert(globals.fifo_hash, name, node);
|
|
|
|
|
2008-04-04 21:18:16 +00:00
|
|
|
node->importance = importance;
|
|
|
|
|
2007-12-18 23:07:48 +00:00
|
|
|
return node;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void send_presence(fifo_node_t *node)
|
|
|
|
{
|
|
|
|
switch_event_t *event;
|
2008-04-04 21:18:16 +00:00
|
|
|
int wait_count = 0;
|
2007-12-18 23:07:48 +00:00
|
|
|
|
2008-01-03 21:34:44 +00:00
|
|
|
if (!globals.running) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2007-12-18 23:07:48 +00:00
|
|
|
if (switch_event_create(&event, SWITCH_EVENT_PRESENCE_IN) == SWITCH_STATUS_SUCCESS) {
|
|
|
|
switch_event_add_header(event, SWITCH_STACK_BOTTOM, "proto", "%s", "park");
|
|
|
|
switch_event_add_header(event, SWITCH_STACK_BOTTOM, "login", "%s", node->name);
|
|
|
|
switch_event_add_header(event, SWITCH_STACK_BOTTOM, "from", "%s", node->name);
|
2008-04-04 21:18:16 +00:00
|
|
|
if ((wait_count = node_consumer_wait_count(node)) > 0) {
|
|
|
|
switch_event_add_header(event, SWITCH_STACK_BOTTOM, "status", "Active (%d waiting)", wait_count);
|
2007-12-18 23:07:48 +00:00
|
|
|
} else {
|
|
|
|
switch_event_add_header(event, SWITCH_STACK_BOTTOM, "status", "Idle");
|
|
|
|
}
|
2008-01-01 22:27:51 +00:00
|
|
|
switch_event_add_header(event, SWITCH_STACK_BOTTOM, "rpid", "%s", "unknown");
|
2007-12-18 23:07:48 +00:00
|
|
|
switch_event_add_header(event, SWITCH_STACK_BOTTOM, "event_type", "presence");
|
|
|
|
switch_event_add_header(event, SWITCH_STACK_BOTTOM, "alt_event_type", "dialog");
|
|
|
|
switch_event_add_header(event, SWITCH_STACK_BOTTOM, "event_count", "%d", 0);
|
|
|
|
|
2008-04-04 21:18:16 +00:00
|
|
|
switch_event_add_header(event, SWITCH_STACK_BOTTOM, "channel-state", "%s", wait_count > 0 ? "CS_RING" : "CS_HANGUP");
|
2007-12-18 23:07:48 +00:00
|
|
|
switch_event_add_header(event, SWITCH_STACK_BOTTOM, "unique-id", "%s", node->name);
|
2008-04-04 21:18:16 +00:00
|
|
|
switch_event_add_header(event, SWITCH_STACK_BOTTOM, "answer-state", "%s", wait_count > 0 ? "early" : "terminated");
|
2007-12-29 19:41:41 +00:00
|
|
|
switch_event_add_header(event, SWITCH_STACK_BOTTOM, "call-direction", "%s", "inbound");
|
2007-12-18 23:07:48 +00:00
|
|
|
switch_event_fire(&event);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static void pres_event_handler(switch_event_t *event)
|
|
|
|
{
|
|
|
|
char *to = switch_event_get_header(event, "to");
|
|
|
|
char *dup_to = NULL, *node_name;
|
|
|
|
fifo_node_t *node;
|
2008-01-03 21:34:44 +00:00
|
|
|
|
|
|
|
if (!globals.running) {
|
|
|
|
return;
|
|
|
|
}
|
2007-12-18 23:07:48 +00:00
|
|
|
|
|
|
|
if (!to || strncasecmp(to, "park+", 5)) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
dup_to = strdup(to);
|
|
|
|
switch_assert(dup_to);
|
|
|
|
|
|
|
|
node_name = dup_to + 5;
|
|
|
|
|
|
|
|
switch_mutex_lock(globals.mutex);
|
|
|
|
if (!(node = switch_core_hash_find(globals.fifo_hash, node_name))) {
|
2008-04-04 21:18:16 +00:00
|
|
|
node = create_node(node_name, 0);
|
2007-12-18 23:07:48 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
switch_mutex_lock(node->mutex);
|
|
|
|
send_presence(node);
|
|
|
|
switch_mutex_unlock(node->mutex);
|
|
|
|
|
|
|
|
switch_mutex_unlock(globals.mutex);
|
|
|
|
|
|
|
|
switch_safe_free(dup_to);
|
|
|
|
}
|
|
|
|
|
2008-04-04 21:37:53 +00:00
|
|
|
typedef enum {
|
|
|
|
STRAT_MORE_PPL,
|
|
|
|
STRAT_WAITING_LONGER,
|
|
|
|
} fifo_strategy_t;
|
|
|
|
|
2008-04-04 21:18:16 +00:00
|
|
|
#define MAX_NODES_PER_CONSUMER 25
|
2007-10-01 20:39:16 +00:00
|
|
|
#define FIFO_DESC "Fifo for stacking parked calls."
|
2007-10-02 00:03:53 +00:00
|
|
|
#define FIFO_USAGE "<fifo name> [in [<announce file>|undef] [<music file>|undef] | out [wait|nowait] [<announce file>|undef] [<music file>|undef]]"
|
2007-10-01 20:39:16 +00:00
|
|
|
SWITCH_STANDARD_APP(fifo_function)
|
|
|
|
{
|
|
|
|
int argc;
|
2007-10-02 00:03:53 +00:00
|
|
|
char *mydata = NULL, *argv[5] = { 0 };
|
2008-04-04 21:18:16 +00:00
|
|
|
fifo_node_t *node = NULL, *node_list[MAX_NODES_PER_CONSUMER+1] = { 0 };
|
2008-01-28 07:26:10 +00:00
|
|
|
switch_channel_t *channel = switch_core_session_get_channel(session);
|
2008-04-04 21:18:16 +00:00
|
|
|
int do_wait = 1, node_count = 0, i = 0;
|
2007-11-01 11:28:26 +00:00
|
|
|
const char *moh = NULL;
|
|
|
|
const char *announce = NULL;
|
2007-10-02 19:58:06 +00:00
|
|
|
switch_event_t *event = NULL;
|
|
|
|
char date[80] = "";
|
|
|
|
switch_time_exp_t tm;
|
2007-10-03 16:44:11 +00:00
|
|
|
switch_time_t ts = switch_timestamp_now();
|
2007-10-02 19:58:06 +00:00
|
|
|
switch_size_t retsize;
|
2008-04-04 21:18:16 +00:00
|
|
|
char *list_string;
|
|
|
|
int nlist_count;
|
|
|
|
char *nlist[MAX_NODES_PER_CONSUMER];
|
|
|
|
int consumer = 0;
|
|
|
|
const char *arg_fifo_name = NULL;
|
|
|
|
const char *arg_inout = NULL;
|
2008-04-03 21:27:16 +00:00
|
|
|
const char *serviced_uuid = NULL;
|
2007-10-02 19:58:06 +00:00
|
|
|
|
2008-01-03 21:34:44 +00:00
|
|
|
if (!globals.running) {
|
|
|
|
return;
|
|
|
|
}
|
2007-10-02 00:03:53 +00:00
|
|
|
|
2008-01-28 07:26:10 +00:00
|
|
|
if (switch_strlen_zero(data)) {
|
2007-10-01 20:39:16 +00:00
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_ERROR, "No Args\n");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
mydata = switch_core_session_strdup(session, data);
|
2007-12-12 03:21:14 +00:00
|
|
|
switch_assert(mydata);
|
2008-04-04 21:18:16 +00:00
|
|
|
|
|
|
|
argc = switch_separate_string(mydata, ' ', argv, (sizeof(argv) / sizeof(argv[0])));
|
|
|
|
arg_fifo_name = argv[0];
|
|
|
|
arg_inout = argv[1];
|
|
|
|
|
2008-04-04 22:02:08 +00:00
|
|
|
if (!(arg_fifo_name && arg_inout)) {
|
2007-10-01 20:39:16 +00:00
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_ERROR, "USAGE %s\n", FIFO_USAGE);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2008-04-04 21:18:16 +00:00
|
|
|
if (!strcasecmp(arg_inout, "out")) {
|
|
|
|
consumer = 1;
|
|
|
|
} else if (strcasecmp(arg_inout, "in")) {
|
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_ERROR, "USAGE %s\n", FIFO_USAGE);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
list_string = switch_core_session_strdup(session, arg_fifo_name);
|
|
|
|
|
|
|
|
if (!(nlist_count = switch_separate_string(list_string, ',', nlist, (sizeof(nlist) / sizeof(nlist[0]))))) {
|
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_ERROR, "USAGE %s\n", FIFO_USAGE);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!consumer && nlist_count > 1) {
|
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_ERROR, "USAGE %s\n", FIFO_USAGE);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2008-01-28 07:26:10 +00:00
|
|
|
switch_mutex_lock(globals.mutex);
|
2008-04-04 21:18:16 +00:00
|
|
|
for(i = 0; i < nlist_count; i++) {
|
|
|
|
int importance = 0;
|
|
|
|
char *p;
|
|
|
|
|
|
|
|
if ((p = strrchr(nlist[i], '!'))) {
|
|
|
|
*p++ = '\0';
|
|
|
|
importance = atoi(p);
|
|
|
|
if (importance < 0) {
|
|
|
|
importance = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!(node = switch_core_hash_find(globals.fifo_hash, nlist[i]))) {
|
|
|
|
node = create_node(nlist[i], importance);
|
|
|
|
}
|
|
|
|
node_list[node_count++] = node;
|
|
|
|
}
|
2007-10-02 19:58:06 +00:00
|
|
|
switch_mutex_unlock(globals.mutex);
|
|
|
|
|
2008-01-28 07:26:10 +00:00
|
|
|
moh = switch_channel_get_variable(channel, "fifo_music");
|
2007-10-02 00:03:53 +00:00
|
|
|
announce = switch_channel_get_variable(channel, "fifo_announce");
|
2008-04-04 21:53:45 +00:00
|
|
|
|
|
|
|
if (consumer) {
|
|
|
|
if (argc > 3) {
|
|
|
|
announce = argv[3];
|
|
|
|
}
|
|
|
|
|
|
|
|
if (argc > 4) {
|
|
|
|
moh = argv[4];
|
|
|
|
}
|
|
|
|
|
|
|
|
} else {
|
|
|
|
if (argc > 2) {
|
|
|
|
announce = argv[2];
|
|
|
|
}
|
|
|
|
|
|
|
|
if (argc > 3) {
|
|
|
|
moh = argv[3];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2008-04-04 21:18:16 +00:00
|
|
|
check_string(announce);
|
|
|
|
check_string(moh);
|
2007-10-01 20:39:16 +00:00
|
|
|
|
2008-04-04 21:18:16 +00:00
|
|
|
if (!consumer && node) {
|
2008-04-03 21:27:16 +00:00
|
|
|
switch_core_session_t *other_session;
|
|
|
|
switch_channel_t *other_channel;
|
2007-11-01 11:28:26 +00:00
|
|
|
const char *uuid = strdup(switch_core_session_get_uuid(session));
|
2008-03-19 16:03:51 +00:00
|
|
|
const char *pri;
|
2008-04-04 21:18:16 +00:00
|
|
|
char tmp[25] = "";
|
2008-03-19 16:03:51 +00:00
|
|
|
int p = 0;
|
2008-04-03 21:27:16 +00:00
|
|
|
int aborted = 0;
|
2008-04-04 21:18:16 +00:00
|
|
|
fifo_chime_data_t cd = { {0} };
|
|
|
|
const char *chime_list = switch_channel_get_variable(channel, "fifo_chime_list");
|
|
|
|
const char *chime_freq = switch_channel_get_variable(channel, "fifo_chime_freq");
|
|
|
|
const char *orbit_var = switch_channel_get_variable(channel, "fifo_orbit_exten");
|
|
|
|
const char *orbit_ann = switch_channel_get_variable(channel, "fifo_orbit_announce");
|
|
|
|
const char *caller_exit_key = switch_channel_get_variable(channel, "fifo_caller_exit_key");
|
|
|
|
int freq = 30;
|
|
|
|
int ftmp = 0;
|
|
|
|
int to = 60;
|
|
|
|
|
|
|
|
if (orbit_var) {
|
|
|
|
char *ot;
|
|
|
|
if ((cd.orbit_exten = switch_core_session_strdup(session, orbit_var))) {
|
|
|
|
if ((ot = strchr(cd.orbit_exten, ':'))) {
|
|
|
|
*ot++ = '\0';
|
|
|
|
if ((to = atoi(ot)) < 0) {
|
|
|
|
to = 60;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
cd.orbit_timeout = switch_timestamp(NULL) + to;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
if (chime_freq) {
|
|
|
|
ftmp = atoi(chime_freq);
|
|
|
|
if (ftmp > 0) {
|
|
|
|
freq = ftmp;
|
|
|
|
}
|
|
|
|
}
|
2007-10-01 20:39:16 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
2008-04-04 21:18:16 +00:00
|
|
|
switch_channel_answer(channel);
|
2007-10-02 00:03:53 +00:00
|
|
|
|
2007-10-02 19:58:06 +00:00
|
|
|
switch_mutex_lock(node->mutex);
|
|
|
|
node->caller_count++;
|
2007-12-18 23:07:48 +00:00
|
|
|
send_presence(node);
|
2007-10-02 19:58:06 +00:00
|
|
|
switch_core_hash_insert(node->caller_hash, uuid, session);
|
2008-03-19 16:03:51 +00:00
|
|
|
|
|
|
|
if ((pri = switch_channel_get_variable(channel, "fifo_priority"))) {
|
|
|
|
p = atoi(pri);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (p >= MAX_PRI) {
|
|
|
|
p = MAX_PRI - 1;
|
|
|
|
}
|
2008-04-04 21:18:16 +00:00
|
|
|
|
|
|
|
if (!node_consumer_wait_count(node)) {
|
|
|
|
node->start_waiting = switch_timestamp_now();
|
|
|
|
}
|
|
|
|
|
2008-03-19 16:03:51 +00:00
|
|
|
switch_queue_push(node->fifo_list[p], (void *)uuid);
|
|
|
|
|
2008-04-04 21:18:16 +00:00
|
|
|
if (!pri) {
|
|
|
|
switch_snprintf(tmp, sizeof(tmp), "%d", p);
|
|
|
|
switch_channel_set_variable(channel, "fifo_priority", tmp);
|
|
|
|
}
|
2007-10-02 19:58:06 +00:00
|
|
|
|
2008-04-04 21:18:16 +00:00
|
|
|
switch_mutex_unlock(node->mutex);
|
|
|
|
|
2007-10-03 16:44:11 +00:00
|
|
|
ts = switch_timestamp_now();
|
2007-10-02 19:58:06 +00:00
|
|
|
switch_time_exp_lt(&tm, ts);
|
|
|
|
switch_strftime(date, &retsize, sizeof(date), "%Y-%m-%d %T", &tm);
|
|
|
|
switch_channel_set_variable(channel, "fifo_status", "WAITING");
|
|
|
|
switch_channel_set_variable(channel, "fifo_timestamp", date);
|
2007-10-02 00:03:53 +00:00
|
|
|
|
2007-10-02 19:58:06 +00:00
|
|
|
if (switch_event_create_subclass(&event, SWITCH_EVENT_CUSTOM, FIFO_EVENT) == SWITCH_STATUS_SUCCESS) {
|
|
|
|
switch_channel_event_set_data(channel, event);
|
|
|
|
switch_event_add_header(event, SWITCH_STACK_BOTTOM, "FIFO-Name", "%s", argv[0]);
|
|
|
|
switch_event_add_header(event, SWITCH_STACK_BOTTOM, "FIFO-Action", "push");
|
2008-04-04 21:18:16 +00:00
|
|
|
switch_event_add_header(event, SWITCH_STACK_BOTTOM, "FIFO-Slot", "%d", p);
|
2007-10-02 19:58:06 +00:00
|
|
|
switch_event_fire(&event);
|
|
|
|
}
|
2008-02-15 16:01:12 +00:00
|
|
|
|
2008-04-03 21:27:16 +00:00
|
|
|
switch_channel_set_flag(channel, CF_TAGGED);
|
|
|
|
|
2008-04-04 21:18:16 +00:00
|
|
|
if (chime_list) {
|
|
|
|
char *list_dup = switch_core_session_strdup(session, chime_list);
|
|
|
|
cd.total = switch_separate_string(list_dup, ',', cd.list, (sizeof(cd.list) / sizeof(cd.list[0])));
|
|
|
|
cd.freq = freq;
|
|
|
|
cd.next = switch_timestamp(NULL) + cd.freq;
|
|
|
|
}
|
|
|
|
|
2008-04-03 21:27:16 +00:00
|
|
|
while(switch_channel_ready(channel)) {
|
|
|
|
switch_input_args_t args = { 0 };
|
|
|
|
char buf[25] = "";
|
2008-04-04 21:18:16 +00:00
|
|
|
|
2008-04-03 21:27:16 +00:00
|
|
|
args.input_callback = moh_on_dtmf;
|
|
|
|
args.buf = buf;
|
|
|
|
args.buflen = sizeof(buf);
|
2008-04-04 21:18:16 +00:00
|
|
|
|
|
|
|
if (cd.total || cd.orbit_timeout) {
|
|
|
|
args.read_frame_callback = caller_read_frame_callback;
|
|
|
|
args.user_data = &cd;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (cd.abort || cd.do_orbit) {
|
|
|
|
aborted = 1;
|
|
|
|
goto abort;
|
|
|
|
}
|
2008-04-03 21:27:16 +00:00
|
|
|
|
|
|
|
if ((serviced_uuid = switch_channel_get_variable(channel, "fifo_serviced_uuid"))) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (moh) {
|
|
|
|
switch_ivr_play_file(session, NULL, moh, &args);
|
|
|
|
} else {
|
|
|
|
switch_ivr_collect_digits_callback(session, &args, 0);
|
|
|
|
}
|
|
|
|
|
2008-04-03 21:31:04 +00:00
|
|
|
if (caller_exit_key && *buf == *caller_exit_key) {
|
2008-04-03 21:27:16 +00:00
|
|
|
aborted = 1;
|
|
|
|
goto abort;
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!serviced_uuid && switch_channel_ready(channel)) {
|
|
|
|
switch_channel_hangup(channel, SWITCH_CAUSE_NORMAL_CLEARING);
|
|
|
|
} else if ((other_session = switch_core_session_locate(serviced_uuid))) {
|
|
|
|
int ready;
|
|
|
|
other_channel = switch_core_session_get_channel(other_session);
|
|
|
|
ready = switch_channel_ready(other_channel);
|
|
|
|
switch_core_session_rwunlock(other_session);
|
|
|
|
if (!ready) {
|
|
|
|
switch_channel_hangup(channel, SWITCH_CAUSE_NORMAL_CLEARING);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2007-10-02 19:58:06 +00:00
|
|
|
|
2008-01-28 07:26:10 +00:00
|
|
|
if (switch_channel_ready(channel)) {
|
2007-10-02 00:03:53 +00:00
|
|
|
if (announce) {
|
|
|
|
switch_ivr_play_file(session, NULL, announce, NULL);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
switch_channel_clear_flag(channel, CF_TAGGED);
|
2008-04-03 21:27:16 +00:00
|
|
|
|
|
|
|
abort:
|
|
|
|
|
|
|
|
if (!aborted && switch_channel_ready(channel)) {
|
2007-10-02 19:58:06 +00:00
|
|
|
switch_channel_set_state(channel, CS_HIBERNATE);
|
|
|
|
} else {
|
2007-10-03 16:44:11 +00:00
|
|
|
ts = switch_timestamp_now();
|
2007-10-02 19:58:06 +00:00
|
|
|
switch_time_exp_lt(&tm, ts);
|
|
|
|
switch_strftime(date, &retsize, sizeof(date), "%Y-%m-%d %T", &tm);
|
2008-04-04 21:18:16 +00:00
|
|
|
switch_channel_set_variable(channel, "fifo_status", cd.do_orbit ? "TIMEOUT" : "ABORTED");
|
2007-10-02 19:58:06 +00:00
|
|
|
switch_channel_set_variable(channel, "fifo_timestamp", date);
|
|
|
|
|
|
|
|
if (switch_event_create_subclass(&event, SWITCH_EVENT_CUSTOM, FIFO_EVENT) == SWITCH_STATUS_SUCCESS) {
|
|
|
|
switch_channel_event_set_data(channel, event);
|
|
|
|
switch_event_add_header(event, SWITCH_STACK_BOTTOM, "FIFO-Name", "%s", argv[0]);
|
2008-04-04 21:18:16 +00:00
|
|
|
switch_event_add_header(event, SWITCH_STACK_BOTTOM, "FIFO-Action", cd.do_orbit ? "timeout" : "abort");
|
2007-10-02 19:58:06 +00:00
|
|
|
switch_event_fire(&event);
|
|
|
|
}
|
|
|
|
switch_mutex_lock(node->mutex);
|
2008-04-04 21:18:16 +00:00
|
|
|
node_remove_uuid(node, uuid);
|
2007-10-02 19:58:06 +00:00
|
|
|
node->caller_count--;
|
2007-12-18 23:07:48 +00:00
|
|
|
send_presence(node);
|
2007-10-02 19:58:06 +00:00
|
|
|
switch_core_hash_delete(node->caller_hash, uuid);
|
|
|
|
switch_mutex_unlock(node->mutex);
|
2007-10-02 00:03:53 +00:00
|
|
|
}
|
|
|
|
|
2008-04-04 21:18:16 +00:00
|
|
|
if (cd.do_orbit && cd.orbit_exten) {
|
|
|
|
if (orbit_ann) {
|
|
|
|
switch_ivr_play_file(session, NULL, orbit_ann, NULL);
|
|
|
|
}
|
|
|
|
switch_ivr_session_transfer(session, cd.orbit_exten, NULL, NULL);
|
|
|
|
}
|
|
|
|
|
2007-10-01 20:39:16 +00:00
|
|
|
return;
|
2008-04-04 21:18:16 +00:00
|
|
|
} else { /* consumer */
|
2008-03-20 00:27:04 +00:00
|
|
|
void *pop = NULL;
|
2007-10-01 20:39:16 +00:00
|
|
|
switch_frame_t *read_frame;
|
|
|
|
switch_status_t status;
|
|
|
|
char *uuid;
|
|
|
|
int done = 0;
|
|
|
|
switch_core_session_t *other_session;
|
2007-10-02 00:03:53 +00:00
|
|
|
switch_input_args_t args = { 0 };
|
2008-03-19 16:03:51 +00:00
|
|
|
const char *pop_order = NULL;
|
|
|
|
int custom_pop = 0;
|
|
|
|
int pop_array[MAX_PRI] = { 0 };
|
|
|
|
char *pop_list[MAX_PRI] = { 0 };
|
2008-03-25 18:48:20 +00:00
|
|
|
const char *fifo_consumer_wrapup_sound = NULL;
|
|
|
|
const char *fifo_consumer_wrapup_key = NULL;
|
2008-04-03 21:27:16 +00:00
|
|
|
const char *my_id;
|
2008-03-25 15:14:05 +00:00
|
|
|
char buf[5] = "";
|
2008-04-04 21:37:53 +00:00
|
|
|
const char *strat_str = switch_channel_get_variable(channel, "fifo_strategy");
|
|
|
|
fifo_strategy_t strat = STRAT_WAITING_LONGER;
|
|
|
|
|
|
|
|
|
|
|
|
if (!switch_strlen_zero(strat_str)) {
|
|
|
|
if (!strcasecmp(strat_str, "more_ppl")) {
|
|
|
|
strat = STRAT_MORE_PPL;
|
|
|
|
} else if (!strcasecmp(strat_str, "waiting_longer")) {
|
|
|
|
strat = STRAT_WAITING_LONGER;
|
|
|
|
} else {
|
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_ERROR, "Invaid strategy\n");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
2008-03-25 15:14:05 +00:00
|
|
|
|
2008-04-03 20:08:39 +00:00
|
|
|
|
|
|
|
if (argc > 2) {
|
|
|
|
if (!strcasecmp(argv[2], "nowait")) {
|
2008-04-04 21:18:16 +00:00
|
|
|
do_wait = 0;
|
2008-04-03 20:08:39 +00:00
|
|
|
} else if (strcasecmp(argv[2], "wait")) {
|
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_ERROR, "USAGE %s\n", FIFO_USAGE);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
2007-10-02 00:03:53 +00:00
|
|
|
|
2008-04-04 21:18:16 +00:00
|
|
|
|
|
|
|
if (!(my_id = switch_channel_get_variable(channel, "fifo_consumer_id"))) {
|
|
|
|
my_id = switch_core_session_get_uuid(session);
|
|
|
|
}
|
|
|
|
|
2007-10-02 19:58:06 +00:00
|
|
|
|
2008-04-04 21:18:16 +00:00
|
|
|
if (do_wait) {
|
|
|
|
for (i = 0; i < node_count; i++) {
|
|
|
|
if (!(node = node_list[i])) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
switch_mutex_lock(node->mutex);
|
|
|
|
node->consumer_count++;
|
|
|
|
switch_core_hash_insert(node->consumer_hash, switch_core_session_get_uuid(session), session);
|
|
|
|
switch_mutex_unlock(node->mutex);
|
|
|
|
}
|
2007-10-01 20:39:16 +00:00
|
|
|
switch_channel_answer(channel);
|
|
|
|
}
|
2008-04-04 21:37:53 +00:00
|
|
|
|
2008-01-28 07:26:10 +00:00
|
|
|
if (switch_event_create_subclass(&event, SWITCH_EVENT_CUSTOM, FIFO_EVENT) == SWITCH_STATUS_SUCCESS) {
|
2007-10-02 19:58:06 +00:00
|
|
|
switch_channel_event_set_data(channel, event);
|
|
|
|
switch_event_add_header(event, SWITCH_STACK_BOTTOM, "FIFO-Name", "%s", argv[0]);
|
|
|
|
switch_event_add_header(event, SWITCH_STACK_BOTTOM, "FIFO-Action", "consumer_start");
|
|
|
|
switch_event_fire(&event);
|
|
|
|
}
|
|
|
|
|
2007-10-03 16:44:11 +00:00
|
|
|
ts = switch_timestamp_now();
|
2007-10-02 19:58:06 +00:00
|
|
|
switch_time_exp_lt(&tm, ts);
|
|
|
|
switch_strftime(date, &retsize, sizeof(date), "%Y-%m-%d %T", &tm);
|
|
|
|
switch_channel_set_variable(channel, "fifo_status", "WAITING");
|
|
|
|
switch_channel_set_variable(channel, "fifo_timestamp", date);
|
|
|
|
|
2008-03-19 16:03:51 +00:00
|
|
|
if ((pop_order = switch_channel_get_variable(channel, "fifo_pop_order"))) {
|
|
|
|
char *tmp = switch_core_session_strdup(session, pop_order);
|
|
|
|
int x;
|
|
|
|
custom_pop = switch_separate_string(tmp, ',', pop_list, (sizeof(pop_list) / sizeof(pop_list[0])));
|
|
|
|
if (custom_pop >= MAX_PRI) {
|
|
|
|
custom_pop = MAX_PRI -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (x = 0; x < custom_pop; x++) {
|
2008-04-04 22:02:08 +00:00
|
|
|
switch_assert(pop_list[x]);
|
2008-03-25 14:29:43 +00:00
|
|
|
int temp = atoi(pop_list[x]);
|
|
|
|
if (temp > -1 && temp < MAX_PRI) {
|
|
|
|
pop_array[x] = temp;
|
2008-03-19 16:03:51 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2008-01-28 07:26:10 +00:00
|
|
|
while(switch_channel_ready(channel)) {
|
2008-04-04 21:18:16 +00:00
|
|
|
int x = 0, winner = -1;
|
|
|
|
switch_time_t longest = 0xFFFFFFFFFFFFFFFF / 2;
|
2008-04-04 21:37:53 +00:00
|
|
|
uint32_t importance = 0, waiting = 0, most_waiting = 0;
|
|
|
|
|
2008-03-25 15:14:05 +00:00
|
|
|
pop = NULL;
|
2008-04-04 15:17:17 +00:00
|
|
|
|
2008-04-04 21:18:16 +00:00
|
|
|
if (moh && do_wait) {
|
2008-03-25 18:48:20 +00:00
|
|
|
memset(&args, 0, sizeof(args));
|
2008-04-04 21:18:16 +00:00
|
|
|
args.read_frame_callback = consumer_read_frame_callback;
|
|
|
|
args.user_data = node_list;
|
2007-10-02 00:03:53 +00:00
|
|
|
switch_ivr_play_file(session, NULL, moh, &args);
|
|
|
|
}
|
2008-03-19 16:03:51 +00:00
|
|
|
|
2008-04-04 21:18:16 +00:00
|
|
|
for(i = 0; i < node_count; i++) {
|
|
|
|
if (!(node = node_list[i])) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2008-04-04 21:37:53 +00:00
|
|
|
if ((waiting = node_consumer_wait_count(node))) {
|
|
|
|
|
|
|
|
if (!importance || node->importance > importance) {
|
|
|
|
if (strat == STRAT_WAITING_LONGER) {
|
|
|
|
if (node->start_waiting < longest) {
|
|
|
|
longest = node->start_waiting;
|
|
|
|
winner = i;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if (waiting > most_waiting) {
|
|
|
|
most_waiting = waiting;
|
|
|
|
winner = i;
|
|
|
|
}
|
|
|
|
}
|
2008-04-04 21:18:16 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (node->importance > importance) {
|
|
|
|
importance = node->importance;
|
2008-03-19 16:03:51 +00:00
|
|
|
}
|
|
|
|
}
|
2008-04-04 21:18:16 +00:00
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
if (winner > -1) {
|
|
|
|
node = node_list[winner];
|
2008-03-19 16:03:51 +00:00
|
|
|
} else {
|
2008-04-04 21:18:16 +00:00
|
|
|
node = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (node) {
|
|
|
|
if (custom_pop) {
|
|
|
|
for(x = 0; x < MAX_PRI; x++) {
|
|
|
|
if (switch_queue_trypop(node->fifo_list[pop_array[x]], &pop) == SWITCH_STATUS_SUCCESS && pop) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
for(x = 0; x < MAX_PRI; x++) {
|
|
|
|
if (switch_queue_trypop(node->fifo_list[x], &pop) == SWITCH_STATUS_SUCCESS && pop) {
|
|
|
|
break;
|
|
|
|
}
|
2008-03-19 16:03:51 +00:00
|
|
|
}
|
|
|
|
}
|
2008-04-04 21:18:16 +00:00
|
|
|
|
|
|
|
if (pop && !node_consumer_wait_count(node)) {
|
|
|
|
switch_mutex_lock(node->mutex);
|
|
|
|
node->start_waiting = 0;
|
|
|
|
switch_mutex_unlock(node->mutex);
|
|
|
|
}
|
2008-03-19 16:03:51 +00:00
|
|
|
}
|
2008-04-04 21:18:16 +00:00
|
|
|
|
2008-03-19 16:03:51 +00:00
|
|
|
|
|
|
|
if (!pop) {
|
2008-04-04 21:18:16 +00:00
|
|
|
if (!do_wait) {
|
2007-10-02 19:58:06 +00:00
|
|
|
break;
|
2007-10-01 20:39:16 +00:00
|
|
|
}
|
2008-03-19 16:03:51 +00:00
|
|
|
|
2007-10-01 20:39:16 +00:00
|
|
|
status = switch_core_session_read_frame(session, &read_frame, -1, 0);
|
2008-03-19 16:03:51 +00:00
|
|
|
|
2007-10-01 20:39:16 +00:00
|
|
|
if (!SWITCH_READ_ACCEPTABLE(status)) {
|
2007-10-02 19:58:06 +00:00
|
|
|
break;
|
2007-10-01 20:39:16 +00:00
|
|
|
}
|
2008-03-19 16:03:51 +00:00
|
|
|
|
2007-10-01 20:39:16 +00:00
|
|
|
continue;
|
|
|
|
}
|
2008-03-19 16:03:51 +00:00
|
|
|
|
2007-10-01 20:39:16 +00:00
|
|
|
uuid = (char *) pop;
|
2008-03-25 15:14:05 +00:00
|
|
|
pop = NULL;
|
2007-10-01 20:39:16 +00:00
|
|
|
|
2008-04-04 21:18:16 +00:00
|
|
|
if (node && (other_session = switch_core_session_locate(uuid))) {
|
2007-10-01 20:39:16 +00:00
|
|
|
switch_channel_t *other_channel = switch_core_session_get_channel(other_session);
|
2007-10-02 00:03:53 +00:00
|
|
|
switch_caller_profile_t *cloned_profile;
|
2008-04-03 21:33:16 +00:00
|
|
|
const char *o_announce = NULL;
|
2007-10-02 19:58:06 +00:00
|
|
|
|
2008-01-28 07:26:10 +00:00
|
|
|
if (switch_event_create_subclass(&event, SWITCH_EVENT_CUSTOM, FIFO_EVENT) == SWITCH_STATUS_SUCCESS) {
|
2007-10-02 19:58:06 +00:00
|
|
|
switch_channel_event_set_data(other_channel, event);
|
|
|
|
switch_event_add_header(event, SWITCH_STACK_BOTTOM, "FIFO-Name", "%s", argv[0]);
|
|
|
|
switch_event_add_header(event, SWITCH_STACK_BOTTOM, "FIFO-Action", "caller_pop");
|
|
|
|
switch_event_fire(&event);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (switch_event_create_subclass(&event, SWITCH_EVENT_CUSTOM, FIFO_EVENT) == SWITCH_STATUS_SUCCESS) {
|
|
|
|
switch_channel_event_set_data(channel, event);
|
|
|
|
switch_event_add_header(event, SWITCH_STACK_BOTTOM, "FIFO-Name", "%s", argv[0]);
|
|
|
|
switch_event_add_header(event, SWITCH_STACK_BOTTOM, "FIFO-Action", "consumer_pop");
|
|
|
|
switch_event_fire(&event);
|
|
|
|
}
|
|
|
|
|
2008-04-03 21:33:16 +00:00
|
|
|
if ((o_announce = switch_channel_get_variable(other_channel, "fifo_override_announce"))) {
|
|
|
|
announce = o_announce;
|
|
|
|
}
|
|
|
|
|
2008-01-28 07:26:10 +00:00
|
|
|
if (announce) {
|
2007-10-02 00:03:53 +00:00
|
|
|
switch_ivr_play_file(session, NULL, announce, NULL);
|
|
|
|
} else {
|
2008-04-03 21:33:16 +00:00
|
|
|
switch_ivr_sleep(session, 500);
|
2007-10-02 00:03:53 +00:00
|
|
|
}
|
2008-04-03 21:27:16 +00:00
|
|
|
|
|
|
|
|
|
|
|
switch_channel_set_variable(other_channel, "fifo_serviced_by", my_id);
|
|
|
|
switch_channel_set_variable(other_channel, "fifo_serviced_uuid", switch_core_session_get_uuid(session));
|
|
|
|
switch_channel_set_flag(other_channel, CF_BREAK);
|
|
|
|
|
|
|
|
while (switch_channel_ready(channel) && switch_channel_ready(other_channel) && switch_channel_test_flag(other_channel, CF_TAGGED)) {
|
|
|
|
status = switch_core_session_read_frame(session, &read_frame, -1, 0);
|
|
|
|
if (!SWITCH_READ_ACCEPTABLE(status)) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!(switch_channel_ready(channel))) {
|
|
|
|
switch_channel_hangup(other_channel, SWITCH_CAUSE_NORMAL_CLEARING);
|
|
|
|
switch_core_session_rwunlock(other_session);
|
|
|
|
break;
|
|
|
|
}
|
2007-10-02 19:58:06 +00:00
|
|
|
|
2007-10-02 00:03:53 +00:00
|
|
|
|
|
|
|
switch_channel_answer(channel);
|
2007-10-02 16:38:15 +00:00
|
|
|
cloned_profile = switch_caller_profile_clone(other_session, switch_channel_get_caller_profile(channel));
|
2007-12-12 03:21:14 +00:00
|
|
|
switch_assert(cloned_profile);
|
2007-10-02 00:03:53 +00:00
|
|
|
switch_channel_set_originator_caller_profile(other_channel, cloned_profile);
|
|
|
|
|
2007-10-02 16:38:15 +00:00
|
|
|
cloned_profile = switch_caller_profile_clone(session, switch_channel_get_caller_profile(other_channel));
|
2007-12-12 03:21:14 +00:00
|
|
|
switch_assert(cloned_profile);
|
|
|
|
switch_assert(cloned_profile->next == NULL);
|
2007-10-02 00:03:53 +00:00
|
|
|
switch_channel_set_originatee_caller_profile(channel, cloned_profile);
|
|
|
|
|
2007-10-03 16:44:11 +00:00
|
|
|
ts = switch_timestamp_now();
|
2007-10-02 19:58:06 +00:00
|
|
|
switch_time_exp_lt(&tm, ts);
|
|
|
|
switch_strftime(date, &retsize, sizeof(date), "%Y-%m-%d %T", &tm);
|
|
|
|
switch_channel_set_variable(channel, "fifo_status", "TALKING");
|
|
|
|
switch_channel_set_variable(channel, "fifo_target", uuid);
|
|
|
|
switch_channel_set_variable(channel, "fifo_timestamp", date);
|
|
|
|
|
|
|
|
switch_channel_set_variable(other_channel, "fifo_status", "TALKING");
|
|
|
|
switch_channel_set_variable(other_channel, "fifo_timestamp", date);
|
|
|
|
switch_channel_set_variable(other_channel, "fifo_target", switch_core_session_get_uuid(session));
|
2007-12-18 23:07:48 +00:00
|
|
|
switch_mutex_lock(node->mutex);
|
|
|
|
send_presence(node);
|
|
|
|
switch_mutex_unlock(node->mutex);
|
2007-10-02 00:03:53 +00:00
|
|
|
switch_ivr_multi_threaded_bridge(session, other_session, on_dtmf, other_session, session);
|
2007-10-02 19:58:06 +00:00
|
|
|
|
2007-10-03 16:44:11 +00:00
|
|
|
ts = switch_timestamp_now();
|
2007-10-02 19:58:06 +00:00
|
|
|
switch_time_exp_lt(&tm, ts);
|
|
|
|
switch_strftime(date, &retsize, sizeof(date), "%Y-%m-%d %T", &tm);
|
|
|
|
switch_channel_set_variable(channel, "fifo_status", "WAITING");
|
|
|
|
switch_channel_set_variable(channel, "fifo_timestamp", date);
|
|
|
|
|
|
|
|
switch_channel_set_variable(other_channel, "fifo_status", "DONE");
|
|
|
|
switch_channel_set_variable(other_channel, "fifo_timestamp", date);
|
|
|
|
|
|
|
|
switch_mutex_lock(node->mutex);
|
|
|
|
node->caller_count--;
|
2007-12-18 23:07:48 +00:00
|
|
|
send_presence(node);
|
2007-10-02 19:58:06 +00:00
|
|
|
switch_core_hash_delete(node->caller_hash, uuid);
|
|
|
|
switch_mutex_unlock(node->mutex);
|
2007-10-01 20:39:16 +00:00
|
|
|
switch_core_session_rwunlock(other_session);
|
2008-04-04 21:18:16 +00:00
|
|
|
if (!do_wait) {
|
2007-10-02 00:03:53 +00:00
|
|
|
done = 1;
|
|
|
|
}
|
2008-03-25 18:48:20 +00:00
|
|
|
|
|
|
|
fifo_consumer_wrapup_sound = switch_channel_get_variable(channel, "fifo_consumer_wrapup_sound");
|
|
|
|
fifo_consumer_wrapup_key = switch_channel_get_variable(channel, "fifo_consumer_wrapup_key");
|
|
|
|
memset(buf, 0, sizeof(buf));
|
|
|
|
|
|
|
|
if (!switch_strlen_zero(fifo_consumer_wrapup_sound)) {
|
|
|
|
memset(&args, 0, sizeof(args));
|
|
|
|
args.buf = buf;
|
|
|
|
args.buflen = sizeof(buf);
|
|
|
|
switch_ivr_play_file(session, NULL, fifo_consumer_wrapup_sound, &args);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!switch_strlen_zero(fifo_consumer_wrapup_key) && strcmp(buf, fifo_consumer_wrapup_key)) {
|
|
|
|
for(;;) {
|
|
|
|
char terminator = 0;
|
|
|
|
switch_ivr_collect_digits_count(session, buf, sizeof(buf)-1, 1, fifo_consumer_wrapup_key, &terminator, 0, 0, 0);
|
|
|
|
if (terminator == *fifo_consumer_wrapup_key) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2008-03-25 15:14:05 +00:00
|
|
|
}
|
|
|
|
}
|
2008-04-04 21:18:16 +00:00
|
|
|
|
2007-10-01 20:39:16 +00:00
|
|
|
switch_safe_free(uuid);
|
|
|
|
|
|
|
|
if (done) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2008-04-04 21:18:16 +00:00
|
|
|
|
2008-01-28 07:26:10 +00:00
|
|
|
if (switch_event_create_subclass(&event, SWITCH_EVENT_CUSTOM, FIFO_EVENT) == SWITCH_STATUS_SUCCESS) {
|
2007-10-02 19:58:06 +00:00
|
|
|
switch_channel_event_set_data(channel, event);
|
|
|
|
switch_event_add_header(event, SWITCH_STACK_BOTTOM, "FIFO-Name", "%s", argv[0]);
|
|
|
|
switch_event_add_header(event, SWITCH_STACK_BOTTOM, "FIFO-Action", "consumer_stop");
|
|
|
|
switch_event_fire(&event);
|
|
|
|
}
|
|
|
|
|
2008-04-04 21:18:16 +00:00
|
|
|
if (do_wait) {
|
|
|
|
for (i = 0; i < node_count; i++) {
|
|
|
|
if (!(node = node_list[i])) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
switch_mutex_lock(node->mutex);
|
|
|
|
switch_core_hash_delete(node->consumer_hash, switch_core_session_get_uuid(session));
|
|
|
|
node->consumer_count--;
|
|
|
|
switch_mutex_unlock(node->mutex);
|
|
|
|
}
|
2007-10-02 19:58:06 +00:00
|
|
|
}
|
2008-04-04 21:18:16 +00:00
|
|
|
|
2007-10-01 20:39:16 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2007-10-02 19:58:06 +00:00
|
|
|
static int xml_hash(switch_xml_t xml, switch_hash_t *hash, char *container, char *tag, int cc_off, int verbose)
|
|
|
|
{
|
|
|
|
switch_xml_t x_tmp, x_caller, x_cp, variables;
|
|
|
|
switch_hash_index_t *hi;
|
|
|
|
switch_core_session_t *session;
|
|
|
|
switch_channel_t *channel;
|
|
|
|
void *val;
|
|
|
|
const void *var;
|
|
|
|
|
|
|
|
x_tmp = switch_xml_add_child_d(xml, container, cc_off++);
|
2007-12-12 03:21:14 +00:00
|
|
|
switch_assert(x_tmp);
|
2007-10-02 19:58:06 +00:00
|
|
|
|
|
|
|
for (hi = switch_hash_first(NULL, hash); hi; hi = switch_hash_next(hi)) {
|
|
|
|
int c_off = 0, d_off = 0;
|
2007-11-01 11:28:26 +00:00
|
|
|
const char *status;
|
|
|
|
const char *ts;
|
2007-10-02 19:58:06 +00:00
|
|
|
|
|
|
|
switch_hash_this(hi, &var, NULL, &val);
|
|
|
|
session = (switch_core_session_t *) val;
|
|
|
|
channel = switch_core_session_get_channel(session);
|
|
|
|
x_caller = switch_xml_add_child_d(x_tmp, tag, c_off++);
|
2007-12-12 03:21:14 +00:00
|
|
|
switch_assert(x_caller);
|
2007-10-02 19:58:06 +00:00
|
|
|
|
|
|
|
switch_xml_set_attr_d(x_caller, "uuid", switch_core_session_get_uuid(session));
|
|
|
|
|
|
|
|
if ((status = switch_channel_get_variable(channel, "fifo_status"))) {
|
|
|
|
switch_xml_set_attr_d(x_caller, "status", status);
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((ts = switch_channel_get_variable(channel, "fifo_timestamp"))) {
|
|
|
|
switch_xml_set_attr_d(x_caller, "timestamp", ts);
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((ts = switch_channel_get_variable(channel, "fifo_target"))) {
|
|
|
|
switch_xml_set_attr_d(x_caller, "target", ts);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!(x_cp = switch_xml_add_child_d(x_caller, "caller_profile", d_off++))) {
|
|
|
|
abort();
|
|
|
|
}
|
|
|
|
if (verbose) {
|
|
|
|
d_off += switch_ivr_set_xml_profile_data(x_cp, switch_channel_get_caller_profile(channel), d_off);
|
|
|
|
|
|
|
|
if (!(variables = switch_xml_add_child_d(x_caller, "variables", c_off++))) {
|
|
|
|
abort();
|
|
|
|
}
|
|
|
|
|
|
|
|
switch_ivr_set_xml_chan_vars(variables, channel, c_off);
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
return cc_off;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void list_node(fifo_node_t *node, switch_xml_t x_report, int *off, int verbose)
|
|
|
|
{
|
|
|
|
|
|
|
|
switch_xml_t x_fifo;
|
|
|
|
int cc_off = 0;
|
2007-11-04 11:05:52 +00:00
|
|
|
char buffer[35];
|
|
|
|
char *tmp = buffer;
|
2007-10-02 19:58:06 +00:00
|
|
|
|
|
|
|
x_fifo = switch_xml_add_child_d(x_report, "fifo", (*off)++);;
|
2007-12-12 03:21:14 +00:00
|
|
|
switch_assert(x_fifo);
|
2007-10-02 19:58:06 +00:00
|
|
|
|
|
|
|
switch_xml_set_attr_d(x_fifo, "name", node->name);
|
2007-12-12 21:53:32 +00:00
|
|
|
switch_snprintf(tmp, sizeof(buffer), "%d", node->consumer_count);
|
2007-10-02 19:58:06 +00:00
|
|
|
switch_xml_set_attr_d(x_fifo, "consumer_count", tmp);
|
2007-12-12 21:53:32 +00:00
|
|
|
switch_snprintf(tmp, sizeof(buffer), "%d", node->caller_count);
|
2007-10-02 19:58:06 +00:00
|
|
|
switch_xml_set_attr_d(x_fifo, "caller_count", tmp);
|
2008-04-04 21:18:16 +00:00
|
|
|
switch_snprintf(tmp, sizeof(buffer), "%d", node_consumer_wait_count(node));
|
2007-12-18 23:07:48 +00:00
|
|
|
switch_xml_set_attr_d(x_fifo, "waiting_count", tmp);
|
2008-04-04 21:18:16 +00:00
|
|
|
switch_snprintf(tmp, sizeof(buffer), "%u", node->importance);
|
|
|
|
switch_xml_set_attr_d(x_fifo, "importance", tmp);
|
2007-10-02 19:58:06 +00:00
|
|
|
|
|
|
|
cc_off = xml_hash(x_fifo, node->caller_hash, "callers", "caller", cc_off, verbose);
|
|
|
|
cc_off = xml_hash(x_fifo, node->consumer_hash, "consumers", "consumer", cc_off, verbose);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2008-04-04 21:18:16 +00:00
|
|
|
#define FIFO_API_SYNTAX "list|list_verbose|count|importance [<fifo name>]"
|
2007-10-02 19:58:06 +00:00
|
|
|
SWITCH_STANDARD_API(fifo_api_function)
|
|
|
|
{
|
|
|
|
int len = 0;
|
|
|
|
fifo_node_t *node;
|
2007-10-29 20:49:22 +00:00
|
|
|
char *data = NULL;
|
2007-10-02 19:58:06 +00:00
|
|
|
int argc = 0;
|
|
|
|
char *argv[5] = { 0 };
|
|
|
|
switch_hash_index_t *hi;
|
|
|
|
void *val;
|
|
|
|
const void *var;
|
|
|
|
int x = 0, verbose = 0;
|
|
|
|
|
2008-01-03 21:34:44 +00:00
|
|
|
|
|
|
|
if (!globals.running) {
|
|
|
|
return SWITCH_STATUS_FALSE;
|
|
|
|
}
|
|
|
|
|
2007-10-02 19:58:06 +00:00
|
|
|
if (!switch_strlen_zero(cmd)) {
|
|
|
|
data = strdup(cmd);
|
2007-12-12 03:21:14 +00:00
|
|
|
switch_assert(data);
|
2007-10-02 19:58:06 +00:00
|
|
|
}
|
|
|
|
|
2007-12-12 03:21:14 +00:00
|
|
|
if (switch_strlen_zero(cmd) || (argc = switch_separate_string(data, ' ', argv, (sizeof(argv) / sizeof(argv[0])))) < 1 || !argv[0]) {
|
2007-10-02 19:58:06 +00:00
|
|
|
stream->write_function(stream, "%s\n", FIFO_API_SYNTAX);
|
|
|
|
return SWITCH_STATUS_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
switch_mutex_lock(globals.mutex);
|
|
|
|
verbose = !strcasecmp(argv[0], "list_verbose");
|
|
|
|
|
|
|
|
if (!strcasecmp(argv[0], "list") || verbose) {
|
|
|
|
char *xml_text = NULL;
|
|
|
|
switch_xml_t x_report = switch_xml_new("fifo_report");
|
2007-12-12 03:21:14 +00:00
|
|
|
switch_assert(x_report);
|
2008-04-04 21:18:16 +00:00
|
|
|
|
2007-10-02 19:58:06 +00:00
|
|
|
if (argc < 2) {
|
|
|
|
for (hi = switch_hash_first(NULL, globals.fifo_hash); hi; hi = switch_hash_next(hi)) {
|
|
|
|
switch_hash_this(hi, &var, NULL, &val);
|
|
|
|
node = (fifo_node_t *) val;
|
|
|
|
switch_mutex_lock(node->mutex);
|
|
|
|
list_node(node, x_report, &x, verbose);
|
|
|
|
switch_mutex_unlock(node->mutex);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if ((node = switch_core_hash_find(globals.fifo_hash, argv[1]))) {
|
|
|
|
switch_mutex_lock(node->mutex);
|
|
|
|
list_node(node, x_report, &x, verbose);
|
|
|
|
switch_mutex_unlock(node->mutex);
|
|
|
|
}
|
|
|
|
}
|
2007-11-30 23:45:27 +00:00
|
|
|
xml_text = switch_xml_toxml(x_report, SWITCH_FALSE);
|
2007-12-12 03:21:14 +00:00
|
|
|
switch_assert(xml_text);
|
2007-10-02 19:58:06 +00:00
|
|
|
stream->write_function(stream, "%s\n", xml_text);
|
|
|
|
switch_xml_free(x_report);
|
|
|
|
switch_safe_free(xml_text);
|
|
|
|
|
2008-04-04 21:18:16 +00:00
|
|
|
} else if (!strcasecmp(argv[0], "importance")) {
|
|
|
|
if ((node = switch_core_hash_find(globals.fifo_hash, argv[1]))) {
|
|
|
|
int importance = 0;
|
|
|
|
if (argc > 2) {
|
|
|
|
importance = atoi(argv[2]);
|
|
|
|
if (importance < 0) {
|
|
|
|
importance = 0;
|
|
|
|
}
|
|
|
|
node->importance = importance;
|
|
|
|
}
|
|
|
|
stream->write_function(stream, "importance: %u\n", node->importance);
|
|
|
|
} else {
|
|
|
|
stream->write_function(stream, "no fifo by that name\n");
|
|
|
|
}
|
2007-10-02 19:58:06 +00:00
|
|
|
} else if (!strcasecmp(argv[0], "count")) {
|
|
|
|
if (argc < 2) {
|
|
|
|
for (hi = switch_hash_first(NULL, globals.fifo_hash); hi; hi = switch_hash_next(hi)) {
|
|
|
|
switch_hash_this(hi, &var, NULL, &val);
|
|
|
|
node = (fifo_node_t *) val;
|
2008-04-04 21:18:16 +00:00
|
|
|
len = node_consumer_wait_count(node);
|
2007-10-02 19:58:06 +00:00
|
|
|
switch_mutex_lock(node->mutex);
|
|
|
|
stream->write_function(stream, "%s:%d:%d:%d\n", (char *)var, node->consumer_count, node->caller_count, len);
|
|
|
|
switch_mutex_unlock(node->mutex);
|
|
|
|
x++;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!x) {
|
|
|
|
stream->write_function(stream, "none\n");
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if ((node = switch_core_hash_find(globals.fifo_hash, argv[1]))) {
|
2008-04-04 21:18:16 +00:00
|
|
|
len = node_consumer_wait_count(node);
|
2007-10-02 19:58:06 +00:00
|
|
|
}
|
|
|
|
switch_mutex_lock(node->mutex);
|
|
|
|
stream->write_function(stream, "%s:%d:%d:%d\n", argv[1], node->consumer_count, node->caller_count, len);
|
|
|
|
switch_mutex_unlock(node->mutex);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
switch_mutex_unlock(globals.mutex);
|
|
|
|
return SWITCH_STATUS_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2007-10-01 20:39:16 +00:00
|
|
|
SWITCH_MODULE_LOAD_FUNCTION(mod_fifo_load)
|
|
|
|
{
|
|
|
|
switch_application_interface_t *app_interface;
|
2007-10-02 19:58:06 +00:00
|
|
|
switch_api_interface_t *commands_api_interface;
|
|
|
|
|
|
|
|
|
|
|
|
/* create/register custom event message type */
|
|
|
|
if (switch_event_reserve_subclass(FIFO_EVENT) != SWITCH_STATUS_SUCCESS) {
|
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_ERROR, "Couldn't register subclass %s!", FIFO_EVENT);
|
|
|
|
return SWITCH_STATUS_TERM;
|
|
|
|
}
|
2007-10-01 20:39:16 +00:00
|
|
|
|
2007-12-18 23:07:48 +00:00
|
|
|
/* Subscribe to presence request events */
|
|
|
|
if (switch_event_bind((char *) modname, SWITCH_EVENT_PRESENCE_PROBE, SWITCH_EVENT_SUBCLASS_ANY, pres_event_handler, NULL) != SWITCH_STATUS_SUCCESS) {
|
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_ERROR, "Couldn't subscribe to presence request events!\n");
|
|
|
|
return SWITCH_STATUS_GENERR;
|
|
|
|
}
|
|
|
|
|
2007-10-01 20:39:16 +00:00
|
|
|
switch_core_new_memory_pool(&globals.pool);
|
|
|
|
switch_core_hash_init(&globals.fifo_hash, globals.pool);
|
2007-10-02 19:58:06 +00:00
|
|
|
switch_mutex_init(&globals.mutex, SWITCH_MUTEX_NESTED, globals.pool);
|
2007-10-01 20:39:16 +00:00
|
|
|
|
|
|
|
/* connect my internal structure to the blank pointer passed to me */
|
|
|
|
*module_interface = switch_loadable_module_create_module_interface(pool, modname);
|
|
|
|
SWITCH_ADD_APP(app_interface, "fifo", "Park with FIFO", FIFO_DESC, fifo_function, FIFO_USAGE, SAF_NONE);
|
2007-10-02 19:58:06 +00:00
|
|
|
SWITCH_ADD_API(commands_api_interface, "fifo", "Return data about a fifo", fifo_api_function, FIFO_API_SYNTAX);
|
2008-01-03 21:34:44 +00:00
|
|
|
globals.running = 1;
|
2007-10-01 20:39:16 +00:00
|
|
|
|
|
|
|
return SWITCH_STATUS_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
Called when the system shuts down
|
|
|
|
*/
|
|
|
|
SWITCH_MODULE_SHUTDOWN_FUNCTION(mod_fifo_shutdown)
|
|
|
|
{
|
|
|
|
switch_hash_index_t *hi;
|
|
|
|
void *val, *pop;
|
2007-10-02 19:58:06 +00:00
|
|
|
fifo_node_t *node;
|
2008-01-03 21:34:44 +00:00
|
|
|
switch_memory_pool_t *pool = globals.pool;
|
2008-01-03 22:04:19 +00:00
|
|
|
switch_mutex_t *mutex = globals.mutex;
|
2008-01-03 21:34:44 +00:00
|
|
|
|
2008-01-03 22:04:19 +00:00
|
|
|
switch_mutex_lock(mutex);
|
|
|
|
|
2008-01-03 21:34:44 +00:00
|
|
|
globals.running = 0;
|
2007-10-01 20:39:16 +00:00
|
|
|
/* Cleanup*/
|
|
|
|
for (hi = switch_hash_first(NULL, globals.fifo_hash); hi; hi = switch_hash_next(hi)) {
|
2008-03-19 16:03:51 +00:00
|
|
|
int x = 0 ;
|
2007-10-01 20:39:16 +00:00
|
|
|
switch_hash_this(hi, NULL, NULL, &val);
|
2007-10-02 19:58:06 +00:00
|
|
|
node = (fifo_node_t *) val;
|
2008-03-19 16:03:51 +00:00
|
|
|
for (x = 0; x < MAX_PRI; x++) {
|
|
|
|
while (switch_queue_trypop(node->fifo_list[x], &pop) == SWITCH_STATUS_SUCCESS) {
|
|
|
|
free(pop);
|
|
|
|
}
|
|
|
|
}
|
2007-10-02 19:58:06 +00:00
|
|
|
switch_core_hash_destroy(&node->caller_hash);
|
|
|
|
switch_core_hash_destroy(&node->consumer_hash);
|
2007-10-01 20:39:16 +00:00
|
|
|
}
|
|
|
|
switch_core_hash_destroy(&globals.fifo_hash);
|
2008-01-03 22:04:19 +00:00
|
|
|
memset(&globals, 0, sizeof(globals));
|
|
|
|
switch_mutex_unlock(mutex);
|
2008-01-03 21:34:44 +00:00
|
|
|
switch_core_destroy_memory_pool(&pool);
|
2007-10-01 20:39:16 +00:00
|
|
|
return SWITCH_STATUS_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/* For Emacs:
|
|
|
|
* Local Variables:
|
|
|
|
* mode:c
|
2008-02-03 22:14:57 +00:00
|
|
|
* indent-tabs-mode:t
|
2007-10-01 20:39:16 +00:00
|
|
|
* tab-width:4
|
|
|
|
* c-basic-offset:4
|
|
|
|
* End:
|
|
|
|
* For VIM:
|
|
|
|
* vim:set softtabstop=4 shiftwidth=4 tabstop=4 expandtab:
|
|
|
|
*/
|