5880 lines
178 KiB
C
5880 lines
178 KiB
C
/*
|
|
* FreeSWITCH Modular Media Switching Software Library / Soft-Switch Application
|
|
* Copyright (C) 2005-2015, Anthony Minessale II <anthm@freeswitch.org>
|
|
*
|
|
* 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 <anthm@freeswitch.org>
|
|
* Portions created by the Initial Developer are Copyright (C)
|
|
* the Initial Developer. All Rights Reserved.
|
|
*
|
|
* Contributor(s):
|
|
*
|
|
* Anthony Minessale II <anthm@freeswitch.org>
|
|
* Michael Jerris <mike@jerris.com>
|
|
* Bret McDanel <bret AT 0xdecafbad dot com>
|
|
* Luke Dashjr <luke@openmethods.com> (OpenMethods, LLC)
|
|
* Christopher M. Rienzo <chris@rienzo.com>
|
|
*
|
|
* switch_ivr_async.c -- IVR Library (async operations)
|
|
*
|
|
*/
|
|
|
|
#include <switch.h>
|
|
#include "private/switch_core_pvt.h"
|
|
#include <speex/speex_preprocess.h>
|
|
#include <speex/speex_echo.h>
|
|
|
|
struct switch_ivr_dmachine_binding {
|
|
char *digits;
|
|
char *repl;
|
|
int first_match;
|
|
char *substituted;
|
|
int32_t key;
|
|
uint8_t rmatch;
|
|
switch_ivr_dmachine_callback_t callback;
|
|
switch_byte_t is_regex;
|
|
switch_byte_t is_priority;
|
|
void *user_data;
|
|
struct switch_ivr_dmachine_binding *next;
|
|
};
|
|
typedef struct switch_ivr_dmachine_binding switch_ivr_dmachine_binding_t;
|
|
|
|
typedef struct {
|
|
switch_ivr_dmachine_binding_t *binding_list;
|
|
switch_ivr_dmachine_binding_t *tail;
|
|
char *name;
|
|
char *terminators;
|
|
} dm_binding_head_t;
|
|
|
|
struct switch_ivr_dmachine {
|
|
switch_memory_pool_t *pool;
|
|
switch_byte_t my_pool;
|
|
char *name;
|
|
uint32_t digit_timeout_ms;
|
|
uint32_t input_timeout_ms;
|
|
switch_hash_t *binding_hash;
|
|
switch_ivr_dmachine_match_t match;
|
|
switch_digit_action_target_t target;
|
|
char digits[DMACHINE_MAX_DIGIT_LEN];
|
|
char last_matching_digits[DMACHINE_MAX_DIGIT_LEN];
|
|
char last_failed_digits[DMACHINE_MAX_DIGIT_LEN];
|
|
uint32_t cur_digit_len;
|
|
uint32_t max_digit_len;
|
|
switch_time_t last_digit_time;
|
|
switch_byte_t is_match;
|
|
switch_ivr_dmachine_callback_t match_callback;
|
|
switch_ivr_dmachine_callback_t nonmatch_callback;
|
|
dm_binding_head_t *realm;
|
|
switch_ivr_dmachine_binding_t *last_matching_binding;
|
|
void *user_data;
|
|
switch_mutex_t *mutex;
|
|
switch_status_t last_return;
|
|
uint8_t pinging;
|
|
};
|
|
|
|
static const char *get_recording_var(switch_channel_t *channel, switch_event_t *vars, const char *name);
|
|
static int recording_var_true(switch_channel_t *channel, switch_event_t *vars, const char *name);
|
|
static switch_status_t speech_on_dtmf(switch_core_session_t *session, const switch_dtmf_t *dtmf, switch_dtmf_direction_t direction);
|
|
|
|
SWITCH_DECLARE(switch_status_t) switch_ivr_dmachine_last_ping(switch_ivr_dmachine_t *dmachine)
|
|
{
|
|
return dmachine->last_return;
|
|
}
|
|
|
|
SWITCH_DECLARE(switch_digit_action_target_t) switch_ivr_dmachine_get_target(switch_ivr_dmachine_t *dmachine)
|
|
{
|
|
switch_assert(dmachine);
|
|
return dmachine->target;
|
|
}
|
|
|
|
SWITCH_DECLARE(void) switch_ivr_dmachine_set_target(switch_ivr_dmachine_t *dmachine, switch_digit_action_target_t target)
|
|
{
|
|
switch_assert(dmachine);
|
|
dmachine->target = target;
|
|
}
|
|
|
|
|
|
SWITCH_DECLARE(void) switch_ivr_dmachine_set_match_callback(switch_ivr_dmachine_t *dmachine, switch_ivr_dmachine_callback_t match_callback)
|
|
{
|
|
|
|
switch_assert(dmachine);
|
|
dmachine->match_callback = match_callback;
|
|
|
|
}
|
|
|
|
SWITCH_DECLARE(void) switch_ivr_dmachine_set_nonmatch_callback(switch_ivr_dmachine_t *dmachine, switch_ivr_dmachine_callback_t nonmatch_callback)
|
|
{
|
|
|
|
switch_assert(dmachine);
|
|
dmachine->nonmatch_callback = nonmatch_callback;
|
|
|
|
}
|
|
|
|
SWITCH_DECLARE(const char *) switch_ivr_dmachine_get_name(switch_ivr_dmachine_t *dmachine)
|
|
{
|
|
return (const char *) dmachine->name;
|
|
}
|
|
|
|
SWITCH_DECLARE(switch_status_t) switch_ivr_dmachine_create(switch_ivr_dmachine_t **dmachine_p,
|
|
const char *name,
|
|
switch_memory_pool_t *pool,
|
|
uint32_t digit_timeout_ms,
|
|
uint32_t input_timeout_ms,
|
|
switch_ivr_dmachine_callback_t match_callback,
|
|
switch_ivr_dmachine_callback_t nonmatch_callback,
|
|
void *user_data)
|
|
{
|
|
switch_byte_t my_pool = 0;
|
|
switch_ivr_dmachine_t *dmachine;
|
|
|
|
if (!pool) {
|
|
switch_core_new_memory_pool(&pool);
|
|
my_pool = 1;
|
|
}
|
|
|
|
dmachine = switch_core_alloc(pool, sizeof(*dmachine));
|
|
dmachine->pool = pool;
|
|
dmachine->my_pool = my_pool;
|
|
dmachine->digit_timeout_ms = digit_timeout_ms;
|
|
dmachine->input_timeout_ms = input_timeout_ms;
|
|
dmachine->match.dmachine = dmachine;
|
|
dmachine->name = switch_core_strdup(dmachine->pool, name);
|
|
switch_mutex_init(&dmachine->mutex, SWITCH_MUTEX_NESTED, dmachine->pool);
|
|
|
|
switch_core_hash_init(&dmachine->binding_hash);
|
|
|
|
if (match_callback) {
|
|
dmachine->match_callback = match_callback;
|
|
}
|
|
|
|
if (nonmatch_callback) {
|
|
dmachine->nonmatch_callback = nonmatch_callback;
|
|
}
|
|
|
|
dmachine->user_data = user_data;
|
|
|
|
*dmachine_p = dmachine;
|
|
|
|
return SWITCH_STATUS_SUCCESS;
|
|
}
|
|
|
|
|
|
SWITCH_DECLARE(void) switch_ivr_dmachine_set_digit_timeout_ms(switch_ivr_dmachine_t *dmachine, uint32_t digit_timeout_ms)
|
|
{
|
|
dmachine->digit_timeout_ms = digit_timeout_ms;
|
|
}
|
|
|
|
SWITCH_DECLARE(void) switch_ivr_dmachine_set_input_timeout_ms(switch_ivr_dmachine_t *dmachine, uint32_t input_timeout_ms)
|
|
{
|
|
dmachine->input_timeout_ms = input_timeout_ms;
|
|
}
|
|
|
|
SWITCH_DECLARE(void) switch_ivr_dmachine_destroy(switch_ivr_dmachine_t **dmachine)
|
|
{
|
|
switch_memory_pool_t *pool;
|
|
|
|
if (!(dmachine && *dmachine)) return;
|
|
|
|
pool = (*dmachine)->pool;
|
|
|
|
switch_core_hash_destroy(&(*dmachine)->binding_hash);
|
|
|
|
if ((*dmachine)->my_pool) {
|
|
switch_core_destroy_memory_pool(&pool);
|
|
}
|
|
}
|
|
|
|
SWITCH_DECLARE(switch_status_t) switch_ivr_dmachine_set_terminators(switch_ivr_dmachine_t *dmachine, const char *terminators)
|
|
{
|
|
if (!dmachine->realm) {
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_ERROR, "No realm selected.\n");
|
|
return SWITCH_STATUS_FALSE;
|
|
}
|
|
|
|
|
|
dmachine->realm->terminators = switch_core_strdup(dmachine->pool, terminators);
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_DEBUG, "Digit parser %s: Setting terminators for realm '%s' to '%s'\n",
|
|
dmachine->name, dmachine->realm->name, terminators);
|
|
|
|
return SWITCH_STATUS_SUCCESS;
|
|
}
|
|
|
|
SWITCH_DECLARE(switch_status_t) switch_ivr_dmachine_set_realm(switch_ivr_dmachine_t *dmachine, const char *realm)
|
|
{
|
|
dm_binding_head_t *headp = switch_core_hash_find(dmachine->binding_hash, realm);
|
|
|
|
if (headp) {
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_INFO, "Digit parser %s: Setting realm to '%s'\n", dmachine->name, realm);
|
|
dmachine->realm = headp;
|
|
return SWITCH_STATUS_SUCCESS;
|
|
}
|
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_ERROR, "Digit parser %s: Error Setting realm to '%s'\n", dmachine->name, realm);
|
|
|
|
return SWITCH_STATUS_FALSE;
|
|
}
|
|
|
|
SWITCH_DECLARE(switch_status_t) switch_ivr_dmachine_clear_realm(switch_ivr_dmachine_t *dmachine, const char *realm)
|
|
{
|
|
dm_binding_head_t *headp;
|
|
|
|
if (zstr(realm)) {
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_ERROR, "Digit parser %s: Error unknown realm: '%s'\n", dmachine->name, realm);
|
|
return SWITCH_STATUS_FALSE;
|
|
}
|
|
|
|
headp = switch_core_hash_find(dmachine->binding_hash, realm);
|
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_INFO, "Digit parser %s: Clearing realm '%s'\n", dmachine->name, realm);
|
|
|
|
if (headp == dmachine->realm) {
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_WARNING,
|
|
"Digit parser %s: '%s' was the active realm, no realm currently selected.\n", dmachine->name, realm);
|
|
dmachine->realm = NULL;
|
|
}
|
|
|
|
/* pool alloc'd just ditch it and it will give back the memory when we destroy ourselves */
|
|
switch_core_hash_delete(dmachine->binding_hash, realm);
|
|
return SWITCH_STATUS_SUCCESS;
|
|
}
|
|
|
|
SWITCH_DECLARE(switch_status_t) switch_ivr_dmachine_bind(switch_ivr_dmachine_t *dmachine,
|
|
const char *realm,
|
|
const char *digits,
|
|
switch_byte_t is_priority,
|
|
int32_t key,
|
|
switch_ivr_dmachine_callback_t callback,
|
|
void *user_data)
|
|
{
|
|
switch_ivr_dmachine_binding_t *binding = NULL, *ptr;
|
|
switch_size_t len;
|
|
dm_binding_head_t *headp;
|
|
const char *msg = "";
|
|
char *repl = NULL;
|
|
char *digits_;
|
|
|
|
if (strlen(digits) > DMACHINE_MAX_DIGIT_LEN -1) {
|
|
return SWITCH_STATUS_FALSE;
|
|
}
|
|
|
|
if (zstr(realm)) {
|
|
realm = "default";
|
|
}
|
|
|
|
if (!(headp = switch_core_hash_find(dmachine->binding_hash, realm))) {
|
|
headp = switch_core_alloc(dmachine->pool, sizeof(*headp));
|
|
headp->name = switch_core_strdup(dmachine->pool, realm);
|
|
switch_core_hash_insert(dmachine->binding_hash, realm, headp);
|
|
}
|
|
|
|
for(ptr = headp->binding_list; ptr; ptr = ptr->next) {
|
|
if ((ptr->is_regex && !strcmp(ptr->digits, digits+1)) || !strcmp(ptr->digits, digits)) {
|
|
msg = "Reuse Existing ";
|
|
binding = ptr;
|
|
binding->callback = callback;
|
|
binding->user_data = user_data;
|
|
goto done;
|
|
}
|
|
}
|
|
|
|
|
|
binding = switch_core_alloc(dmachine->pool, sizeof(*binding));
|
|
|
|
digits_ = switch_core_strdup(dmachine->pool, digits);
|
|
|
|
if (*digits_ == '=') {
|
|
binding->first_match = 1;
|
|
digits_++;
|
|
}
|
|
|
|
if (*digits_ == '~') {
|
|
binding->is_regex = 1;
|
|
digits_++;
|
|
if ((repl = strchr(digits_, '~')) && *(repl+1) == '~') {
|
|
*repl++ = '\0';
|
|
*repl++ = '\0';
|
|
}
|
|
}
|
|
|
|
binding->key = key;
|
|
binding->digits = digits_;
|
|
binding->is_priority = is_priority;
|
|
binding->callback = callback;
|
|
binding->user_data = user_data;
|
|
binding->repl = repl;
|
|
|
|
if (headp->tail) {
|
|
headp->tail->next = binding;
|
|
} else {
|
|
headp->binding_list = binding;
|
|
}
|
|
|
|
headp->tail = binding;
|
|
|
|
len = strlen(digits);
|
|
|
|
if (dmachine->realm != headp) {
|
|
switch_ivr_dmachine_set_realm(dmachine, realm);
|
|
}
|
|
|
|
if (binding->is_regex && dmachine->max_digit_len != DMACHINE_MAX_DIGIT_LEN -1) {
|
|
dmachine->max_digit_len = DMACHINE_MAX_DIGIT_LEN -1;
|
|
} else if (len > dmachine->max_digit_len) {
|
|
dmachine->max_digit_len = (uint32_t) len;
|
|
}
|
|
|
|
done:
|
|
|
|
if (binding->is_regex) {
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_DEBUG, "%sDigit parser %s: binding %s/%s/%d callback: %p data: %p\n",
|
|
msg, dmachine->name, digits, realm, key, (void *)(intptr_t) callback, user_data);
|
|
} else {
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_DEBUG, "%sDigit parser %s: binding %s/%s/%d callback: %p data: %p\n",
|
|
msg, dmachine->name, digits, realm, key, (void *)(intptr_t) callback, user_data);
|
|
}
|
|
|
|
return SWITCH_STATUS_SUCCESS;
|
|
}
|
|
|
|
typedef enum {
|
|
DM_MATCH_NONE,
|
|
DM_MATCH_EXACT,
|
|
DM_MATCH_PARTIAL,
|
|
DM_MATCH_BOTH,
|
|
DM_MATCH_NEVER
|
|
} dm_match_t;
|
|
|
|
|
|
static dm_match_t switch_ivr_dmachine_check_match(switch_ivr_dmachine_t *dmachine, switch_bool_t is_timeout)
|
|
{
|
|
dm_match_t best = DM_MATCH_NONE;
|
|
switch_ivr_dmachine_binding_t *bp, *exact_bp = NULL, *partial_bp = NULL, *both_bp = NULL, *r_bp = NULL;
|
|
int pmatches = 0, ematches = 0, rmatches = 0;
|
|
|
|
if (!dmachine->cur_digit_len || !dmachine->realm) goto end;
|
|
|
|
for(bp = dmachine->realm->binding_list; bp; bp = bp->next) {
|
|
if (bp->is_regex) {
|
|
if (bp->repl) {
|
|
int ovector[30] = { 0 };
|
|
int proceed = 0;
|
|
switch_regex_t *re = NULL;
|
|
|
|
|
|
proceed = switch_regex_perform(dmachine->digits, bp->digits, &re, ovector, sizeof(ovector) / sizeof(ovector[0]));
|
|
|
|
if (proceed) {
|
|
char *substituted = NULL;
|
|
switch_size_t len;
|
|
|
|
len = (strlen(dmachine->digits) + strlen(bp->digits) + 10) * proceed;
|
|
substituted = malloc(len);
|
|
switch_assert(substituted);
|
|
memset(substituted, 0, len);
|
|
switch_perform_substitution(re, proceed, bp->repl, dmachine->digits, substituted, len, ovector);
|
|
|
|
if (!bp->substituted || strcmp(substituted, bp->substituted)) {
|
|
bp->substituted = switch_core_strdup(dmachine->pool, substituted);
|
|
}
|
|
free(substituted);
|
|
switch_regex_safe_free(re);
|
|
bp->rmatch = 1;
|
|
} else {
|
|
bp->substituted = NULL;
|
|
bp->rmatch = 0;
|
|
}
|
|
} else {
|
|
switch_status_t r_status = switch_regex_match(dmachine->digits, bp->digits);
|
|
bp->rmatch = r_status == SWITCH_STATUS_SUCCESS;
|
|
}
|
|
|
|
rmatches++;
|
|
pmatches++;
|
|
|
|
if (bp->rmatch && bp->first_match) break;
|
|
|
|
} else {
|
|
if (!strncmp(dmachine->digits, bp->digits, strlen(dmachine->digits))) {
|
|
pmatches++;
|
|
if (dmachine->cur_digit_len == strlen(bp->digits)) {
|
|
ematches++;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
if (!zstr(dmachine->realm->terminators)) {
|
|
char *p = dmachine->realm->terminators;
|
|
char *q;
|
|
|
|
while(p && *p) {
|
|
if ((q=strrchr(dmachine->digits, *p))) {
|
|
*q = '\0';
|
|
is_timeout = 1;
|
|
break;
|
|
}
|
|
p++;
|
|
}
|
|
}
|
|
|
|
for(bp = dmachine->realm->binding_list; bp; bp = bp->next) {
|
|
if (bp->is_regex) {
|
|
if (bp->rmatch) {
|
|
if (bp->first_match || (bp->is_priority && ! ematches) || is_timeout || (bp == dmachine->realm->binding_list && !bp->next)) {
|
|
best = DM_MATCH_EXACT;
|
|
exact_bp = bp;
|
|
break;
|
|
}
|
|
best = DM_MATCH_PARTIAL;
|
|
}
|
|
} else {
|
|
int pmatch = !strncmp(dmachine->digits, bp->digits, strlen(dmachine->digits));
|
|
|
|
if (!exact_bp && pmatch && (bp->first_match || !rmatches || bp->is_priority || is_timeout) && !strcmp(bp->digits, dmachine->digits)) {
|
|
best = DM_MATCH_EXACT;
|
|
exact_bp = bp;
|
|
if (bp->first_match || bp->is_priority || dmachine->cur_digit_len == dmachine->max_digit_len) break;
|
|
}
|
|
|
|
if (!(both_bp && partial_bp) && strlen(bp->digits) != strlen(dmachine->digits) && pmatch) {
|
|
|
|
if (exact_bp) {
|
|
best = DM_MATCH_BOTH;
|
|
both_bp = bp;
|
|
} else {
|
|
best = DM_MATCH_PARTIAL;
|
|
partial_bp = bp;
|
|
}
|
|
}
|
|
|
|
if (both_bp && exact_bp && partial_bp) break;
|
|
}
|
|
}
|
|
|
|
if (!pmatches) {
|
|
best = DM_MATCH_NEVER;
|
|
}
|
|
|
|
|
|
end:
|
|
|
|
if (is_timeout) {
|
|
if (both_bp) {
|
|
r_bp = exact_bp ? exact_bp : both_bp;
|
|
}
|
|
}
|
|
|
|
if (best == DM_MATCH_EXACT && exact_bp) {
|
|
r_bp = exact_bp;
|
|
}
|
|
|
|
|
|
if (r_bp) {
|
|
dmachine->last_matching_binding = r_bp;
|
|
|
|
if (r_bp->substituted) {
|
|
switch_set_string(dmachine->last_matching_digits, r_bp->substituted);
|
|
} else {
|
|
switch_set_string(dmachine->last_matching_digits, dmachine->digits);
|
|
}
|
|
best = DM_MATCH_EXACT;
|
|
}
|
|
|
|
return best;
|
|
|
|
}
|
|
|
|
static switch_bool_t switch_ivr_dmachine_check_timeout(switch_ivr_dmachine_t *dmachine)
|
|
{
|
|
switch_time_t now = switch_time_now();
|
|
uint32_t timeout = dmachine->cur_digit_len ? dmachine->digit_timeout_ms : dmachine->input_timeout_ms;
|
|
|
|
if (!dmachine->last_digit_time) dmachine->last_digit_time = now;
|
|
|
|
if (timeout) {
|
|
if ((uint32_t)((now - dmachine->last_digit_time) / 1000) > timeout) {
|
|
return SWITCH_TRUE;
|
|
}
|
|
}
|
|
|
|
return SWITCH_FALSE;
|
|
}
|
|
|
|
SWITCH_DECLARE(switch_ivr_dmachine_match_t *) switch_ivr_dmachine_get_match(switch_ivr_dmachine_t *dmachine)
|
|
{
|
|
if (dmachine->is_match) {
|
|
dmachine->is_match = 0;
|
|
return &dmachine->match;
|
|
}
|
|
|
|
return NULL;
|
|
}
|
|
|
|
SWITCH_DECLARE(const char *) switch_ivr_dmachine_get_failed_digits(switch_ivr_dmachine_t *dmachine)
|
|
{
|
|
|
|
return dmachine->last_failed_digits;
|
|
}
|
|
|
|
SWITCH_DECLARE(switch_status_t) switch_ivr_dmachine_ping(switch_ivr_dmachine_t *dmachine, switch_ivr_dmachine_match_t **match_p)
|
|
{
|
|
switch_bool_t is_timeout = switch_ivr_dmachine_check_timeout(dmachine);
|
|
dm_match_t is_match = switch_ivr_dmachine_check_match(dmachine, is_timeout);
|
|
switch_status_t r, s;
|
|
int clear = 0;
|
|
|
|
if (is_match == DM_MATCH_NEVER) {
|
|
is_timeout++;
|
|
}
|
|
|
|
if (switch_mutex_trylock(dmachine->mutex) != SWITCH_STATUS_SUCCESS) {
|
|
return SWITCH_STATUS_SUCCESS;
|
|
}
|
|
|
|
if (dmachine->pinging) {
|
|
switch_mutex_unlock(dmachine->mutex);
|
|
return SWITCH_STATUS_BREAK;
|
|
}
|
|
|
|
dmachine->pinging = 1;
|
|
|
|
if (zstr(dmachine->digits) && !is_timeout) {
|
|
r = SWITCH_STATUS_SUCCESS;
|
|
} else if (dmachine->cur_digit_len > dmachine->max_digit_len) {
|
|
r = SWITCH_STATUS_FALSE;
|
|
} else if (is_match == DM_MATCH_EXACT || (is_match == DM_MATCH_BOTH && is_timeout)) {
|
|
r = SWITCH_STATUS_FOUND;
|
|
|
|
dmachine->match.match_digits = dmachine->last_matching_digits;
|
|
dmachine->match.match_key = dmachine->last_matching_binding->key;
|
|
dmachine->match.user_data = dmachine->last_matching_binding->user_data;
|
|
|
|
if (match_p) {
|
|
*match_p = &dmachine->match;
|
|
}
|
|
|
|
dmachine->is_match = 1;
|
|
|
|
dmachine->match.type = DM_MATCH_POSITIVE;
|
|
|
|
if (dmachine->last_matching_binding->callback) {
|
|
s = dmachine->last_matching_binding->callback(&dmachine->match);
|
|
|
|
switch(s) {
|
|
case SWITCH_STATUS_CONTINUE:
|
|
r = SWITCH_STATUS_SUCCESS;
|
|
break;
|
|
case SWITCH_STATUS_SUCCESS:
|
|
break;
|
|
default:
|
|
r = SWITCH_STATUS_BREAK;
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (dmachine->match_callback) {
|
|
dmachine->match.user_data = dmachine->user_data;
|
|
s = dmachine->match_callback(&dmachine->match);
|
|
|
|
switch(s) {
|
|
case SWITCH_STATUS_CONTINUE:
|
|
r = SWITCH_STATUS_SUCCESS;
|
|
break;
|
|
case SWITCH_STATUS_SUCCESS:
|
|
break;
|
|
default:
|
|
r = SWITCH_STATUS_BREAK;
|
|
break;
|
|
}
|
|
|
|
}
|
|
|
|
clear++;
|
|
} else if (is_timeout) {
|
|
r = SWITCH_STATUS_TIMEOUT;
|
|
} else if (is_match == DM_MATCH_NONE && dmachine->cur_digit_len == dmachine->max_digit_len) {
|
|
r = SWITCH_STATUS_NOTFOUND;
|
|
} else {
|
|
r = SWITCH_STATUS_SUCCESS;
|
|
}
|
|
|
|
if (r != SWITCH_STATUS_FOUND && r != SWITCH_STATUS_SUCCESS && r != SWITCH_STATUS_BREAK) {
|
|
switch_set_string(dmachine->last_failed_digits, dmachine->digits);
|
|
dmachine->match.match_digits = dmachine->last_failed_digits;
|
|
|
|
dmachine->match.type = DM_MATCH_NEGATIVE;
|
|
|
|
if (dmachine->nonmatch_callback) {
|
|
dmachine->match.user_data = dmachine->user_data;
|
|
s = dmachine->nonmatch_callback(&dmachine->match);
|
|
|
|
switch(s) {
|
|
case SWITCH_STATUS_CONTINUE:
|
|
r = SWITCH_STATUS_SUCCESS;
|
|
break;
|
|
case SWITCH_STATUS_SUCCESS:
|
|
break;
|
|
default:
|
|
r = SWITCH_STATUS_BREAK;
|
|
break;
|
|
}
|
|
|
|
}
|
|
|
|
clear++;
|
|
}
|
|
|
|
if (clear) {
|
|
switch_ivr_dmachine_clear(dmachine);
|
|
}
|
|
|
|
dmachine->last_return = r;
|
|
|
|
dmachine->pinging = 0;
|
|
|
|
switch_mutex_unlock(dmachine->mutex);
|
|
|
|
return r;
|
|
}
|
|
|
|
SWITCH_DECLARE(switch_status_t) switch_ivr_dmachine_feed(switch_ivr_dmachine_t *dmachine, const char *digits, switch_ivr_dmachine_match_t **match)
|
|
{
|
|
const char *p;
|
|
switch_status_t status = SWITCH_STATUS_BREAK;
|
|
|
|
if (!zstr(digits)) {
|
|
status = SWITCH_STATUS_SUCCESS;
|
|
}
|
|
|
|
for (p = digits; p && *p; p++) {
|
|
switch_mutex_lock(dmachine->mutex);
|
|
if (dmachine->cur_digit_len < dmachine->max_digit_len) {
|
|
switch_status_t istatus;
|
|
char *e = dmachine->digits + strlen(dmachine->digits);
|
|
|
|
*e++ = *p;
|
|
*e = '\0';
|
|
dmachine->cur_digit_len++;
|
|
switch_mutex_unlock(dmachine->mutex);
|
|
dmachine->last_digit_time = switch_time_now();
|
|
if (status == SWITCH_STATUS_SUCCESS && (istatus = switch_ivr_dmachine_ping(dmachine, match)) != SWITCH_STATUS_SUCCESS) {
|
|
status = istatus;
|
|
}
|
|
} else {
|
|
switch_mutex_unlock(dmachine->mutex);
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_ERROR, "dmachine overflow error!\n");
|
|
status = SWITCH_STATUS_FALSE;
|
|
}
|
|
}
|
|
|
|
return status;
|
|
}
|
|
|
|
SWITCH_DECLARE(switch_bool_t) switch_ivr_dmachine_is_parsing(switch_ivr_dmachine_t *dmachine)
|
|
{
|
|
return !!dmachine->cur_digit_len;
|
|
}
|
|
|
|
SWITCH_DECLARE(switch_status_t) switch_ivr_dmachine_clear(switch_ivr_dmachine_t *dmachine)
|
|
{
|
|
|
|
memset(dmachine->digits, 0, sizeof(dmachine->digits));
|
|
dmachine->cur_digit_len = 0;
|
|
dmachine->last_digit_time = 0;
|
|
return SWITCH_STATUS_SUCCESS;
|
|
}
|
|
|
|
|
|
|
|
SWITCH_DECLARE(switch_status_t) switch_ivr_session_echo(switch_core_session_t *session, switch_input_args_t *args)
|
|
{
|
|
switch_status_t status;
|
|
switch_frame_t *read_frame;
|
|
switch_channel_t *channel = switch_core_session_get_channel(session);
|
|
|
|
if (switch_channel_pre_answer(channel) != SWITCH_STATUS_SUCCESS) {
|
|
return SWITCH_STATUS_FALSE;
|
|
}
|
|
|
|
arg_recursion_check_start(args);
|
|
|
|
if (switch_channel_var_true(channel, "echo_decode_video")) {
|
|
switch_channel_set_flag_recursive(channel, CF_VIDEO_DECODED_READ);
|
|
}
|
|
|
|
if (switch_channel_var_true(channel, "echo_decode_audio")) {
|
|
switch_core_session_raw_read(session);
|
|
}
|
|
|
|
switch_channel_set_flag(channel, CF_VIDEO_ECHO);
|
|
switch_channel_set_flag(channel, CF_TEXT_ECHO);
|
|
|
|
while (switch_channel_ready(channel)) {
|
|
status = switch_core_session_read_frame(session, &read_frame, SWITCH_IO_FLAG_NONE, 0);
|
|
if (!SWITCH_READ_ACCEPTABLE(status)) {
|
|
break;
|
|
}
|
|
|
|
switch_ivr_parse_all_events(session);
|
|
|
|
if (args && (args->input_callback || args->buf || args->buflen)) {
|
|
switch_dtmf_t dtmf = {0};
|
|
|
|
/*
|
|
dtmf handler function you can hook up to be executed when a digit is dialed during playback
|
|
if you return anything but SWITCH_STATUS_SUCCESS the playback will stop.
|
|
*/
|
|
if (switch_channel_has_dtmf(channel)) {
|
|
if (!args->input_callback && !args->buf) {
|
|
break;
|
|
}
|
|
switch_channel_dequeue_dtmf(channel, &dtmf);
|
|
if (args->input_callback) {
|
|
status = args->input_callback(session, (void *) &dtmf, SWITCH_INPUT_TYPE_DTMF, args->buf, args->buflen);
|
|
} else {
|
|
*((char *) args->buf) = dtmf.digit;
|
|
status = SWITCH_STATUS_BREAK;
|
|
}
|
|
}
|
|
|
|
if (args->input_callback) {
|
|
switch_event_t *event = NULL;
|
|
|
|
if (switch_core_session_dequeue_event(session, &event, SWITCH_FALSE) == SWITCH_STATUS_SUCCESS) {
|
|
status = args->input_callback(session, event, SWITCH_INPUT_TYPE_EVENT, args->buf, args->buflen);
|
|
switch_event_destroy(&event);
|
|
}
|
|
}
|
|
|
|
if (status != SWITCH_STATUS_SUCCESS) {
|
|
break;
|
|
}
|
|
}
|
|
|
|
switch_core_session_write_frame(session, read_frame, SWITCH_IO_FLAG_NONE, 0);
|
|
|
|
if (switch_channel_test_flag(channel, CF_BREAK)) {
|
|
switch_channel_clear_flag(channel, CF_BREAK);
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (switch_channel_var_true(channel, "echo_decode_video")) {
|
|
switch_channel_clear_flag_recursive(channel, CF_VIDEO_DECODED_READ);
|
|
}
|
|
|
|
switch_core_session_video_reset(session);
|
|
switch_core_session_reset(session, SWITCH_TRUE, SWITCH_TRUE);
|
|
switch_channel_clear_flag(channel, CF_TEXT_ECHO);
|
|
|
|
return SWITCH_STATUS_SUCCESS;
|
|
}
|
|
|
|
typedef struct {
|
|
switch_file_handle_t fh;
|
|
int mux;
|
|
int loop;
|
|
char *file;
|
|
switch_buffer_t *wbuffer; // only in r&w mode
|
|
switch_mutex_t *mutex;
|
|
} displace_helper_t;
|
|
|
|
static switch_bool_t write_displace_callback(switch_media_bug_t *bug, void *user_data, switch_abc_type_t type)
|
|
{
|
|
displace_helper_t *dh = (displace_helper_t *) user_data;
|
|
|
|
switch (type) {
|
|
case SWITCH_ABC_TYPE_INIT:
|
|
break;
|
|
case SWITCH_ABC_TYPE_CLOSE:
|
|
if (dh) {
|
|
switch_core_session_t *session = switch_core_media_bug_get_session(bug);
|
|
switch_channel_t *channel;
|
|
|
|
switch_core_file_close(&dh->fh);
|
|
|
|
if (session && (channel = switch_core_session_get_channel(session))) {
|
|
switch_channel_set_private(channel, dh->file, NULL);
|
|
}
|
|
}
|
|
break;
|
|
case SWITCH_ABC_TYPE_READ_REPLACE:
|
|
{
|
|
switch_frame_t *rframe = switch_core_media_bug_get_read_replace_frame(bug);
|
|
if (dh && !dh->mux) {
|
|
memset(rframe->data, 255, rframe->datalen);
|
|
}
|
|
switch_core_media_bug_set_read_replace_frame(bug, rframe);
|
|
}
|
|
break;
|
|
case SWITCH_ABC_TYPE_WRITE_REPLACE:
|
|
if (dh) {
|
|
switch_frame_t *rframe = NULL;
|
|
switch_size_t len;
|
|
switch_status_t st;
|
|
|
|
rframe = switch_core_media_bug_get_write_replace_frame(bug);
|
|
len = rframe->samples;
|
|
|
|
if (dh->mux) {
|
|
int16_t buf[SWITCH_RECOMMENDED_BUFFER_SIZE];
|
|
int16_t *fp = rframe->data;
|
|
uint32_t x;
|
|
|
|
st = switch_core_file_read(&dh->fh, buf, &len);
|
|
|
|
for (x = 0; x < (uint32_t) len * dh->fh.channels; x++) {
|
|
int32_t mixed = fp[x] + buf[x];
|
|
switch_normalize_to_16bit(mixed);
|
|
fp[x] = (int16_t) mixed;
|
|
}
|
|
} else {
|
|
st = switch_core_file_read(&dh->fh, rframe->data, &len);
|
|
if (len < rframe->samples) {
|
|
memset((char *)rframe->data + (len * 2 * dh->fh.channels), 0, (rframe->samples - len) * 2 * dh->fh.channels);
|
|
}
|
|
}
|
|
|
|
rframe->datalen = rframe->samples * 2 * dh->fh.channels;
|
|
|
|
if (st != SWITCH_STATUS_SUCCESS || len == 0) {
|
|
if (dh->loop) {
|
|
uint32_t pos = 0;
|
|
switch_core_file_seek(&dh->fh, &pos, 0, SEEK_SET);
|
|
} else {
|
|
switch_core_session_t *session = switch_core_media_bug_get_session(bug);
|
|
switch_channel_t *channel;
|
|
|
|
if (session && (channel = switch_core_session_get_channel(session))) {
|
|
switch_channel_set_private(channel, dh->file, NULL);
|
|
}
|
|
return SWITCH_FALSE;
|
|
}
|
|
}
|
|
|
|
switch_core_media_bug_set_write_replace_frame(bug, rframe);
|
|
}
|
|
break;
|
|
case SWITCH_ABC_TYPE_WRITE:
|
|
default:
|
|
break;
|
|
}
|
|
|
|
return SWITCH_TRUE;
|
|
}
|
|
|
|
static switch_bool_t read_displace_callback(switch_media_bug_t *bug, void *user_data, switch_abc_type_t type)
|
|
{
|
|
displace_helper_t *dh = (displace_helper_t *) user_data;
|
|
|
|
switch (type) {
|
|
case SWITCH_ABC_TYPE_INIT:
|
|
break;
|
|
case SWITCH_ABC_TYPE_CLOSE:
|
|
if (dh) {
|
|
switch_core_session_t *session = switch_core_media_bug_get_session(bug);
|
|
switch_channel_t *channel;
|
|
|
|
if (dh->wbuffer) switch_buffer_destroy(&dh->wbuffer);
|
|
if (dh->mutex) switch_mutex_destroy(dh->mutex);
|
|
|
|
switch_core_file_close(&dh->fh);
|
|
|
|
if (session && (channel = switch_core_session_get_channel(session))) {
|
|
switch_channel_set_private(channel, dh->file, NULL);
|
|
}
|
|
}
|
|
break;
|
|
case SWITCH_ABC_TYPE_WRITE_REPLACE:
|
|
{
|
|
switch_frame_t *rframe = switch_core_media_bug_get_write_replace_frame(bug);
|
|
|
|
if (dh) {
|
|
if (dh->wbuffer) {
|
|
switch_mutex_lock(dh->mutex);
|
|
if (switch_buffer_inuse(dh->wbuffer) >= rframe->datalen) {
|
|
switch_buffer_read(dh->wbuffer, rframe->data, rframe->datalen);
|
|
} else {
|
|
switch_log_printf(SWITCH_CHANNEL_SESSION_LOG(switch_core_media_bug_get_session(bug)),
|
|
SWITCH_LOG_WARNING, "not enough data %" SWITCH_SIZE_T_FMT "\n",
|
|
switch_buffer_inuse(dh->wbuffer));
|
|
memset(rframe->data, 255, rframe->datalen);
|
|
}
|
|
switch_mutex_unlock(dh->mutex);
|
|
} else if (!dh->mux) {
|
|
memset(rframe->data, 255, rframe->datalen);
|
|
}
|
|
}
|
|
|
|
switch_core_media_bug_set_write_replace_frame(bug, rframe);
|
|
}
|
|
break;
|
|
case SWITCH_ABC_TYPE_READ_REPLACE:
|
|
if (dh) {
|
|
switch_frame_t *rframe = NULL;
|
|
switch_size_t len;
|
|
switch_status_t st;
|
|
rframe = switch_core_media_bug_get_read_replace_frame(bug);
|
|
len = rframe->samples;
|
|
|
|
if (dh->mux) {
|
|
int16_t buf[SWITCH_RECOMMENDED_BUFFER_SIZE];
|
|
int16_t *fp = rframe->data;
|
|
uint32_t x;
|
|
|
|
st = switch_core_file_read(&dh->fh, buf, &len);
|
|
|
|
for (x = 0; x < (uint32_t) len * dh->fh.channels; x++) {
|
|
int32_t mixed = fp[x] + buf[x];
|
|
switch_normalize_to_16bit(mixed);
|
|
fp[x] = (int16_t) mixed;
|
|
}
|
|
|
|
} else {
|
|
st = switch_core_file_read(&dh->fh, rframe->data, &len);
|
|
rframe->samples = (uint32_t) len;
|
|
|
|
if (dh->wbuffer) {
|
|
switch_mutex_lock(dh->mutex);
|
|
switch_buffer_write(dh->wbuffer, rframe->data, len * 2 * dh->fh.channels);
|
|
switch_mutex_unlock(dh->mutex);
|
|
}
|
|
}
|
|
|
|
rframe->datalen = rframe->samples * 2 * dh->fh.channels;
|
|
|
|
|
|
if (st != SWITCH_STATUS_SUCCESS || len == 0) {
|
|
if (dh->loop) {
|
|
uint32_t pos = 0;
|
|
switch_core_file_seek(&dh->fh, &pos, 0, SEEK_SET);
|
|
} else {
|
|
switch_core_session_t *session = switch_core_media_bug_get_session(bug);
|
|
switch_channel_t *channel;
|
|
|
|
if (session && (channel = switch_core_session_get_channel(session))) {
|
|
switch_channel_set_private(channel, dh->file, NULL);
|
|
}
|
|
return SWITCH_FALSE;
|
|
}
|
|
}
|
|
|
|
switch_core_media_bug_set_read_replace_frame(bug, rframe);
|
|
}
|
|
break;
|
|
case SWITCH_ABC_TYPE_WRITE:
|
|
default:
|
|
break;
|
|
}
|
|
|
|
return SWITCH_TRUE;
|
|
}
|
|
|
|
SWITCH_DECLARE(switch_status_t) switch_ivr_stop_displace_session(switch_core_session_t *session, const char *file)
|
|
{
|
|
switch_media_bug_t *bug;
|
|
switch_channel_t *channel = switch_core_session_get_channel(session);
|
|
|
|
if ((bug = switch_channel_get_private(channel, file))) {
|
|
switch_channel_set_private(channel, file, NULL);
|
|
switch_core_media_bug_remove(session, &bug);
|
|
return SWITCH_STATUS_SUCCESS;
|
|
}
|
|
|
|
return SWITCH_STATUS_FALSE;
|
|
}
|
|
|
|
SWITCH_DECLARE(switch_status_t) switch_ivr_displace_session(switch_core_session_t *session, const char *file, uint32_t limit, const char *flags)
|
|
{
|
|
switch_channel_t *channel = switch_core_session_get_channel(session);
|
|
switch_media_bug_t *bug;
|
|
switch_status_t status;
|
|
time_t to = 0;
|
|
char *ext;
|
|
const char *prefix;
|
|
displace_helper_t *dh;
|
|
const char *p;
|
|
switch_bool_t hangup_on_error = SWITCH_FALSE;
|
|
switch_media_bug_flag_enum_t bug_flags = 0;
|
|
switch_codec_implementation_t read_impl = { 0 };
|
|
switch_core_session_get_read_impl(session, &read_impl);
|
|
|
|
if ((p = switch_channel_get_variable(channel, "DISPLACE_HANGUP_ON_ERROR"))) {
|
|
hangup_on_error = switch_true(p);
|
|
}
|
|
|
|
if (zstr(file)) {
|
|
return SWITCH_STATUS_FALSE;
|
|
}
|
|
|
|
if (switch_channel_pre_answer(channel) != SWITCH_STATUS_SUCCESS) {
|
|
return SWITCH_STATUS_FALSE;
|
|
}
|
|
|
|
if (!switch_channel_media_up(channel) || !switch_core_session_get_read_codec(session)) {
|
|
switch_log_printf(SWITCH_CHANNEL_SESSION_LOG(session), SWITCH_LOG_ERROR, "Can not displace session. Media not enabled on channel\n");
|
|
return SWITCH_STATUS_FALSE;
|
|
}
|
|
|
|
if ((bug = switch_channel_get_private(channel, file))) {
|
|
switch_log_printf(SWITCH_CHANNEL_SESSION_LOG(session), SWITCH_LOG_ERROR, "Only 1 of the same file per channel please!\n");
|
|
return SWITCH_STATUS_FALSE;
|
|
}
|
|
|
|
if (!(dh = switch_core_session_alloc(session, sizeof(*dh)))) {
|
|
return SWITCH_STATUS_MEMERR;
|
|
}
|
|
|
|
if (!(prefix = switch_channel_get_variable(channel, "sound_prefix"))) {
|
|
prefix = SWITCH_GLOBAL_dirs.base_dir;
|
|
}
|
|
|
|
if (!strstr(file, SWITCH_URL_SEPARATOR)) {
|
|
if (!switch_is_file_path(file)) {
|
|
char *tfile = NULL;
|
|
char *e;
|
|
|
|
if (*file == '[') {
|
|
tfile = switch_core_session_strdup(session, file);
|
|
if ((e = switch_find_end_paren(tfile, '[', ']'))) {
|
|
*e = '\0';
|
|
file = e + 1;
|
|
} else {
|
|
tfile = NULL;
|
|
}
|
|
}
|
|
|
|
file = switch_core_session_sprintf(session, "%s%s%s%s%s", switch_str_nil(tfile), tfile ? "]" : "", prefix, SWITCH_PATH_SEPARATOR, file);
|
|
}
|
|
if ((ext = strrchr(file, '.'))) {
|
|
ext++;
|
|
} else {
|
|
ext = read_impl.iananame;
|
|
file = switch_core_session_sprintf(session, "%s.%s", file, ext);
|
|
}
|
|
}
|
|
|
|
dh->fh.channels = read_impl.number_of_channels;
|
|
dh->fh.samplerate = read_impl.actual_samples_per_second;
|
|
dh->file = switch_core_session_strdup(session, file);
|
|
|
|
if (switch_core_file_open(&dh->fh,
|
|
file,
|
|
read_impl.number_of_channels,
|
|
read_impl.actual_samples_per_second, SWITCH_FILE_FLAG_READ | SWITCH_FILE_DATA_SHORT, NULL) != SWITCH_STATUS_SUCCESS) {
|
|
if (hangup_on_error) {
|
|
switch_channel_hangup(channel, SWITCH_CAUSE_DESTINATION_OUT_OF_ORDER);
|
|
switch_core_session_reset(session, SWITCH_TRUE, SWITCH_TRUE);
|
|
}
|
|
return SWITCH_STATUS_GENERR;
|
|
}
|
|
|
|
if (limit) {
|
|
to = switch_epoch_time_now(NULL) + limit;
|
|
}
|
|
|
|
if (flags && strchr(flags, 'm')) {
|
|
dh->mux++;
|
|
}
|
|
|
|
if (flags && strchr(flags, 'l')) {
|
|
dh->loop++;
|
|
}
|
|
|
|
if (flags && strchr(flags, 'f')) {
|
|
bug_flags |= SMBF_FIRST;
|
|
}
|
|
|
|
if (flags && strchr(flags, 'r')) {
|
|
if (strchr(flags, 'w')) { // r&w mode, both sides can hear the same file
|
|
int len = dh->fh.samplerate / 10 * 2 * dh->fh.channels; // init with 100ms
|
|
|
|
switch_mutex_init(&dh->mutex, SWITCH_MUTEX_NESTED, switch_core_session_get_pool(session));
|
|
switch_buffer_create_dynamic(&dh->wbuffer, len, len, 0);
|
|
}
|
|
|
|
status = switch_core_media_bug_add(session, "displace", file,
|
|
read_displace_callback, dh, to, bug_flags | SMBF_WRITE_REPLACE | SMBF_READ_REPLACE | SMBF_NO_PAUSE, &bug);
|
|
} else {
|
|
status = switch_core_media_bug_add(session, "displace", file,
|
|
write_displace_callback, dh, to, bug_flags | SMBF_WRITE_REPLACE | SMBF_READ_REPLACE | SMBF_NO_PAUSE, &bug);
|
|
}
|
|
|
|
if (status != SWITCH_STATUS_SUCCESS) {
|
|
switch_core_file_close(&dh->fh);
|
|
return status;
|
|
}
|
|
|
|
switch_channel_set_private(channel, file, bug);
|
|
|
|
return SWITCH_STATUS_SUCCESS;
|
|
}
|
|
|
|
|
|
struct record_helper {
|
|
switch_media_bug_t *bug;
|
|
switch_memory_pool_t *helper_pool;
|
|
switch_core_session_t *recording_session;
|
|
switch_core_session_t *transfer_from_session;
|
|
uint8_t transfer_complete;
|
|
char *file;
|
|
switch_file_handle_t *fh;
|
|
switch_file_handle_t in_fh;
|
|
switch_file_handle_t out_fh;
|
|
int native;
|
|
uint32_t packet_len;
|
|
int min_sec;
|
|
int final_timeout_ms;
|
|
int initial_timeout_ms;
|
|
int silence_threshold;
|
|
int silence_timeout_ms;
|
|
switch_time_t silence_time;
|
|
int rready;
|
|
int wready;
|
|
switch_time_t last_read_time;
|
|
switch_time_t last_write_time;
|
|
switch_bool_t hangup_on_error;
|
|
switch_codec_implementation_t read_impl;
|
|
switch_bool_t speech_detected;
|
|
switch_buffer_t *thread_buffer;
|
|
switch_thread_t *thread;
|
|
switch_mutex_t *buffer_mutex;
|
|
int thread_ready;
|
|
uint8_t thread_needs_transfer;
|
|
uint32_t writes;
|
|
uint32_t vwrites;
|
|
const char *completion_cause;
|
|
int start_event_sent;
|
|
switch_event_t *variables;
|
|
switch_mutex_t *cond_mutex;
|
|
switch_thread_cond_t *cond;
|
|
};
|
|
|
|
static switch_status_t record_helper_destroy(struct record_helper **rh, switch_core_session_t *session);
|
|
|
|
/**
|
|
* Set the recording completion cause. The cause can only be set once, to minimize the logic in the record_callback.
|
|
* [The completion_cause strings are essentially those of an MRCP Recorder resource.]
|
|
*/
|
|
static void set_completion_cause(struct record_helper *rh, const char *completion_cause)
|
|
{
|
|
if (!rh->completion_cause) {
|
|
rh->completion_cause = completion_cause;
|
|
}
|
|
}
|
|
|
|
static switch_bool_t is_silence_frame(switch_frame_t *frame, int silence_threshold, switch_codec_implementation_t *codec_impl)
|
|
{
|
|
int16_t *fdata = (int16_t *) frame->data;
|
|
uint32_t samples = frame->datalen / sizeof(*fdata);
|
|
switch_bool_t is_silence = SWITCH_TRUE;
|
|
uint32_t channel_num = 0;
|
|
|
|
int divisor = 0;
|
|
if (!(divisor = codec_impl->samples_per_second / 8000)) {
|
|
divisor = 1;
|
|
}
|
|
|
|
/* is silence only if every channel is silent */
|
|
for (channel_num = 0; channel_num < codec_impl->number_of_channels && is_silence; channel_num++) {
|
|
uint32_t count = 0, j = channel_num;
|
|
double energy = 0;
|
|
for (count = 0; count < samples; count++) {
|
|
energy += abs(fdata[j]);
|
|
j += codec_impl->number_of_channels;
|
|
}
|
|
is_silence &= (uint32_t) ((energy / (samples / divisor)) < silence_threshold);
|
|
}
|
|
|
|
return is_silence;
|
|
}
|
|
|
|
static void merge_recording_variables(struct record_helper *rh, switch_event_t *event)
|
|
{
|
|
switch_event_header_t *hi;
|
|
if (rh->variables) {
|
|
for (hi = rh->variables->headers; hi; hi = hi->next) {
|
|
char buf[1024];
|
|
char *vvar = NULL, *vval = NULL;
|
|
|
|
vvar = (char *) hi->name;
|
|
vval = (char *) hi->value;
|
|
|
|
switch_assert(vvar && vval);
|
|
switch_snprintf(buf, sizeof(buf), "Recording-Variable-%s", vvar);
|
|
switch_event_add_header_string(event, SWITCH_STACK_BOTTOM, buf, vval);
|
|
}
|
|
}
|
|
}
|
|
|
|
static void send_record_stop_event(switch_channel_t *channel, switch_codec_implementation_t *read_impl, struct record_helper *rh)
|
|
{
|
|
switch_event_t *event;
|
|
|
|
if (rh->fh) {
|
|
switch_channel_set_variable_printf(channel, "record_samples", "%d", rh->fh->samples_out);
|
|
if (read_impl->actual_samples_per_second) {
|
|
switch_channel_set_variable_printf(channel, "record_seconds", "%d", rh->fh->samples_out / read_impl->actual_samples_per_second);
|
|
switch_channel_set_variable_printf(channel, "record_ms", "%d", rh->fh->samples_out / (read_impl->actual_samples_per_second / 1000));
|
|
}
|
|
}
|
|
|
|
if (!zstr(rh->completion_cause)) {
|
|
switch_channel_set_variable_printf(channel, "record_completion_cause", "%s", rh->completion_cause);
|
|
}
|
|
|
|
if (switch_event_create(&event, SWITCH_EVENT_RECORD_STOP) == SWITCH_STATUS_SUCCESS) {
|
|
switch_channel_event_set_data(channel, event);
|
|
switch_event_add_header_string(event, SWITCH_STACK_BOTTOM, "Record-File-Path", rh->file);
|
|
merge_recording_variables(rh, event);
|
|
if (!zstr(rh->completion_cause)) {
|
|
switch_event_add_header_string(event, SWITCH_STACK_BOTTOM, "Record-Completion-Cause", rh->completion_cause);
|
|
}
|
|
switch_event_fire(&event);
|
|
}
|
|
|
|
if (rh->start_event_sent) {
|
|
if (rh->variables) {
|
|
const char *exec_app = NULL;
|
|
exec_app = switch_event_get_header(rh->variables, "execute_on_record_stop");
|
|
if (exec_app) {
|
|
switch_channel_execute_on_value(channel, exec_app);
|
|
}
|
|
}
|
|
switch_channel_execute_on(channel, "execute_on_record_stop");
|
|
switch_channel_api_on(channel, "api_on_record_stop");
|
|
}
|
|
|
|
rh->start_event_sent = 0;
|
|
}
|
|
|
|
static void *SWITCH_THREAD_FUNC recording_thread(switch_thread_t *thread, void *obj)
|
|
{
|
|
switch_media_bug_t *bug = (switch_media_bug_t *) obj;
|
|
switch_core_session_t *session = switch_core_media_bug_get_session(bug);
|
|
switch_channel_t *channel = switch_core_session_get_channel(session);
|
|
struct record_helper *rh;
|
|
switch_size_t bsize = SWITCH_RECOMMENDED_BUFFER_SIZE, samples = 0, inuse = 0;
|
|
unsigned char *data;
|
|
int channels = 1;
|
|
switch_codec_implementation_t read_impl = { 0 };
|
|
|
|
if (switch_core_session_read_lock(session) != SWITCH_STATUS_SUCCESS) {
|
|
return NULL;
|
|
}
|
|
|
|
rh = switch_core_media_bug_get_user_data(bug);
|
|
switch_buffer_create_dynamic(&rh->thread_buffer, SWITCH_RECOMMENDED_BUFFER_SIZE, SWITCH_RECOMMENDED_BUFFER_SIZE, 0);
|
|
rh->thread_ready = 1;
|
|
|
|
channels = switch_core_media_bug_test_flag(bug, SMBF_STEREO) ? 2 : rh->read_impl.number_of_channels;
|
|
data = switch_core_alloc(rh->helper_pool, SWITCH_RECOMMENDED_BUFFER_SIZE);
|
|
|
|
switch_mutex_lock(rh->cond_mutex);
|
|
|
|
while(switch_test_flag(rh->fh, SWITCH_FILE_OPEN)) {
|
|
if (rh->thread_needs_transfer) {
|
|
assert(session != rh->recording_session);
|
|
|
|
if (switch_core_session_read_lock(rh->recording_session) != SWITCH_STATUS_SUCCESS) {
|
|
/* Wait until recording is reverted to the original session */
|
|
switch_thread_cond_wait(rh->cond, rh->cond_mutex);
|
|
continue;
|
|
}
|
|
|
|
switch_core_session_rwunlock(session);
|
|
session = rh->recording_session;
|
|
channel = switch_core_session_get_channel(session);
|
|
bug = rh->bug;
|
|
channels = switch_core_media_bug_test_flag(bug, SMBF_STEREO) ? 2 : rh->read_impl.number_of_channels;
|
|
|
|
/* Tell record_callback that we transferred */
|
|
rh->thread_needs_transfer = 0;
|
|
|
|
/* Erasing the obj variable for safety because we transferred (we are under another bug) */
|
|
obj = NULL;
|
|
}
|
|
|
|
if (switch_core_file_has_video(rh->fh, SWITCH_TRUE)) {
|
|
switch_core_session_get_read_impl(session, &read_impl);
|
|
if (read_impl.decoded_bytes_per_packet > 0 && read_impl.decoded_bytes_per_packet <= SWITCH_RECOMMENDED_BUFFER_SIZE) {
|
|
bsize = read_impl.decoded_bytes_per_packet;
|
|
}
|
|
}
|
|
|
|
switch_mutex_lock(rh->buffer_mutex);
|
|
inuse = switch_buffer_inuse(rh->thread_buffer);
|
|
|
|
if ((!rh->thread_ready || switch_channel_down_nosig(channel)) && !inuse) {
|
|
switch_mutex_unlock(rh->buffer_mutex);
|
|
break;
|
|
}
|
|
|
|
if (!inuse) {
|
|
switch_mutex_unlock(rh->buffer_mutex);
|
|
if (rh->thread_ready) {
|
|
switch_thread_cond_wait(rh->cond, rh->cond_mutex);
|
|
}
|
|
continue;
|
|
}
|
|
|
|
samples = switch_buffer_read(rh->thread_buffer, data, bsize) / 2 / channels;
|
|
switch_mutex_unlock(rh->buffer_mutex);
|
|
|
|
if (switch_core_file_write(rh->fh, data, &samples) != SWITCH_STATUS_SUCCESS) {
|
|
switch_log_printf(SWITCH_CHANNEL_SESSION_LOG(session), SWITCH_LOG_ERROR, "Error writing %s\n", rh->file);
|
|
/* File write failed */
|
|
set_completion_cause(rh, "uri-failure");
|
|
if (rh->hangup_on_error) {
|
|
switch_channel_hangup(channel, SWITCH_CAUSE_DESTINATION_OUT_OF_ORDER);
|
|
switch_core_session_reset(session, SWITCH_TRUE, SWITCH_TRUE);
|
|
}
|
|
}
|
|
}
|
|
|
|
switch_mutex_unlock(rh->cond_mutex);
|
|
|
|
switch_core_session_rwunlock(session);
|
|
|
|
return NULL;
|
|
}
|
|
|
|
static void record_helper_post_process(struct record_helper *rh, switch_core_session_t *session)
|
|
{
|
|
switch_channel_t *channel = switch_core_session_get_channel(session);
|
|
const char *var = NULL;
|
|
const char *post_process_exec = NULL;
|
|
if (rh->variables && (post_process_exec = switch_event_get_header(rh->variables, SWITCH_RECORD_POST_PROCESS_EXEC_APP_VARIABLE))) {
|
|
switch_channel_execute_on_value(channel, post_process_exec);
|
|
}
|
|
switch_channel_execute_on(channel, SWITCH_RECORD_POST_PROCESS_EXEC_APP_VARIABLE);
|
|
|
|
if ((var = switch_channel_get_variable(channel, SWITCH_RECORD_POST_PROCESS_EXEC_API_VARIABLE))) {
|
|
char *cmd = switch_core_session_strdup(session, var);
|
|
char *data, *expanded = NULL;
|
|
switch_stream_handle_t stream = { 0 };
|
|
|
|
SWITCH_STANDARD_STREAM(stream);
|
|
|
|
if ((data = strchr(cmd, ':'))) {
|
|
*data++ = '\0';
|
|
expanded = switch_channel_expand_variables(channel, data);
|
|
}
|
|
|
|
switch_api_execute(cmd, expanded, session, &stream);
|
|
|
|
if (expanded && expanded != data) {
|
|
free(expanded);
|
|
}
|
|
|
|
switch_safe_free(stream.data);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
static switch_bool_t record_callback(switch_media_bug_t *bug, void *user_data, switch_abc_type_t type)
|
|
{
|
|
switch_core_session_t *session = switch_core_media_bug_get_session(bug);
|
|
switch_channel_t *channel = switch_core_session_get_channel(session);
|
|
struct record_helper *rh = (struct record_helper *) user_data;
|
|
switch_event_t *event;
|
|
switch_frame_t *nframe;
|
|
switch_size_t len = 0;
|
|
int mask = switch_core_media_bug_test_flag(bug, SMBF_MASK);
|
|
unsigned char null_data[SWITCH_RECOMMENDED_BUFFER_SIZE] = {0};
|
|
|
|
/* Check if the recording was transferred (see recording_follow_transfer) */
|
|
if (rh->recording_session != session) {
|
|
return SWITCH_FALSE;
|
|
}
|
|
|
|
switch (type) {
|
|
case SWITCH_ABC_TYPE_INIT:
|
|
{
|
|
const char *var = get_recording_var(channel, rh->variables, "RECORD_USE_THREAD");
|
|
|
|
switch_core_session_get_read_impl(session, &rh->read_impl);
|
|
|
|
/* Check if recording is transferred from another session */
|
|
if (rh->transfer_from_session && rh->transfer_from_session != rh->recording_session) {
|
|
|
|
/* If there is a thread, we need to re-initiate it */
|
|
if (rh->thread_ready) {
|
|
switch_media_bug_t *oldbug = rh->bug;
|
|
int sanity = 200;
|
|
|
|
switch_log_printf(SWITCH_CHANNEL_SESSION_LOG(session), SWITCH_LOG_DEBUG, "Re-initiating recording thread for file %s\n", rh->file);
|
|
|
|
rh->bug = bug;
|
|
rh->thread_needs_transfer = 1;
|
|
|
|
switch_mutex_lock(rh->cond_mutex);
|
|
switch_thread_cond_signal(rh->cond);
|
|
switch_mutex_unlock(rh->cond_mutex);
|
|
|
|
while (--sanity > 0 && rh->thread_needs_transfer) {
|
|
switch_yield(10000);
|
|
}
|
|
|
|
/* Check if the thread reacted on the transfer */
|
|
if (rh->thread_needs_transfer) {
|
|
/* Thread did not react, assuming it is cond_wait'ing */
|
|
rh->bug = oldbug;
|
|
switch_core_session_get_read_impl(rh->transfer_from_session, &rh->read_impl);
|
|
rh->thread_needs_transfer = 0;
|
|
|
|
return SWITCH_FALSE;
|
|
}
|
|
}
|
|
|
|
if (rh->fh) {
|
|
switch_log_printf(SWITCH_CHANNEL_SESSION_LOG(session), SWITCH_LOG_DEBUG, "Record session sample rate: %d -> %d\n", rh->fh->native_rate, rh->fh->samplerate);
|
|
rh->fh->native_rate = rh->read_impl.actual_samples_per_second;
|
|
if (switch_core_file_has_video(rh->fh, SWITCH_TRUE)) {
|
|
switch_core_media_bug_set_media_params(bug, &rh->fh->mm);
|
|
}
|
|
}
|
|
|
|
rh->transfer_from_session = NULL;
|
|
rh->transfer_complete = 1;
|
|
|
|
break;
|
|
}
|
|
|
|
/* Required for potential record_transfer */
|
|
rh->bug = bug;
|
|
|
|
if (!rh->native && rh->fh && (zstr(var) || switch_true(var))) {
|
|
switch_threadattr_t *thd_attr = NULL;
|
|
int sanity = 200;
|
|
|
|
switch_mutex_init(&rh->buffer_mutex, SWITCH_MUTEX_NESTED, rh->helper_pool);
|
|
switch_mutex_init(&rh->cond_mutex, SWITCH_MUTEX_NESTED, rh->helper_pool);
|
|
switch_thread_cond_create(&rh->cond, rh->helper_pool);
|
|
switch_threadattr_create(&thd_attr, rh->helper_pool);
|
|
switch_threadattr_stacksize_set(thd_attr, SWITCH_THREAD_STACKSIZE);
|
|
switch_threadattr_priority_set(thd_attr, SWITCH_PRI_LOW);
|
|
switch_thread_create(&rh->thread, thd_attr, recording_thread, bug, rh->helper_pool);
|
|
|
|
while(--sanity > 0 && !rh->thread_ready) {
|
|
switch_yield(10000);
|
|
}
|
|
}
|
|
|
|
if(rh->start_event_sent == 0) {
|
|
rh->start_event_sent = 1;
|
|
if (switch_event_create(&event, SWITCH_EVENT_RECORD_START) == SWITCH_STATUS_SUCCESS) {
|
|
switch_channel_event_set_data(channel, event);
|
|
switch_event_add_header_string(event, SWITCH_STACK_BOTTOM, "Record-File-Path", rh->file);
|
|
merge_recording_variables(rh, event);
|
|
switch_event_fire(&event);
|
|
}
|
|
if (rh->variables) {
|
|
const char *exec_app = NULL;
|
|
exec_app = switch_event_get_header(rh->variables, "execute_on_record_start");
|
|
if (exec_app) {
|
|
switch_channel_execute_on_value(channel, exec_app);
|
|
}
|
|
}
|
|
switch_channel_execute_on(channel, "execute_on_record_start");
|
|
switch_channel_api_on(channel, "api_on_record_start");
|
|
}
|
|
|
|
rh->silence_time = switch_micro_time_now();
|
|
rh->silence_timeout_ms = rh->initial_timeout_ms;
|
|
rh->speech_detected = SWITCH_FALSE;
|
|
rh->completion_cause = NULL;
|
|
|
|
if (rh->fh) {
|
|
switch_log_printf(SWITCH_CHANNEL_SESSION_LOG(session), SWITCH_LOG_DEBUG, "Record session sample rate: %d -> %d\n", rh->fh->native_rate, rh->fh->samplerate);
|
|
rh->fh->native_rate = rh->read_impl.actual_samples_per_second;
|
|
if (switch_core_file_has_video(rh->fh, SWITCH_TRUE)) {
|
|
switch_core_media_bug_set_media_params(bug, &rh->fh->mm);
|
|
switch_core_session_request_video_refresh(session);
|
|
switch_core_media_gen_key_frame(session);
|
|
}
|
|
}
|
|
}
|
|
break;
|
|
case SWITCH_ABC_TYPE_TAP_NATIVE_READ:
|
|
{
|
|
switch_time_t now = switch_time_now();
|
|
switch_time_t diff;
|
|
|
|
rh->rready = 1;
|
|
|
|
nframe = switch_core_media_bug_get_native_read_frame(bug);
|
|
len = nframe->datalen;
|
|
|
|
if (!rh->wready) {
|
|
unsigned char fill_data[SWITCH_RECOMMENDED_BUFFER_SIZE] = {0};
|
|
switch_size_t fill_len = len;
|
|
|
|
switch_core_gen_encoded_silence(fill_data, &rh->read_impl, len);
|
|
switch_core_file_write(&rh->out_fh, fill_data, &fill_len);
|
|
}
|
|
|
|
|
|
if (rh->last_read_time && rh->last_read_time < now) {
|
|
diff = (now - rh->last_read_time) / rh->read_impl.microseconds_per_packet;
|
|
|
|
if (diff > 3) {
|
|
unsigned char fill_data[SWITCH_RECOMMENDED_BUFFER_SIZE] = {0};
|
|
switch_core_gen_encoded_silence(fill_data, &rh->read_impl, len);
|
|
|
|
while(diff > 1) {
|
|
switch_size_t fill_len = len;
|
|
switch_core_file_write(&rh->in_fh, fill_data, &fill_len);
|
|
diff--;
|
|
}
|
|
}
|
|
}
|
|
|
|
switch_core_file_write(&rh->in_fh, mask ? null_data : nframe->data, &len);
|
|
rh->last_read_time = now;
|
|
rh->writes++;
|
|
}
|
|
break;
|
|
case SWITCH_ABC_TYPE_TAP_NATIVE_WRITE:
|
|
{
|
|
switch_time_t now = switch_time_now();
|
|
switch_time_t diff;
|
|
rh->wready = 1;
|
|
|
|
nframe = switch_core_media_bug_get_native_write_frame(bug);
|
|
len = nframe->datalen;
|
|
|
|
if (!rh->rready) {
|
|
unsigned char fill_data[SWITCH_RECOMMENDED_BUFFER_SIZE] = {0};
|
|
switch_size_t fill_len = len;
|
|
switch_core_gen_encoded_silence(fill_data, &rh->read_impl, len);
|
|
switch_core_file_write(&rh->in_fh, fill_data, &fill_len);
|
|
}
|
|
|
|
|
|
|
|
|
|
if (rh->last_write_time && rh->last_write_time < now) {
|
|
diff = (now - rh->last_write_time) / rh->read_impl.microseconds_per_packet;
|
|
|
|
if (diff > 3) {
|
|
unsigned char fill_data[SWITCH_RECOMMENDED_BUFFER_SIZE] = {0};
|
|
switch_core_gen_encoded_silence(fill_data, &rh->read_impl, len);
|
|
|
|
while(diff > 1) {
|
|
switch_size_t fill_len = len;
|
|
switch_core_file_write(&rh->out_fh, fill_data, &fill_len);
|
|
diff--;
|
|
}
|
|
}
|
|
}
|
|
|
|
switch_core_file_write(&rh->out_fh, mask ? null_data : nframe->data, &len);
|
|
rh->last_write_time = now;
|
|
rh->writes++;
|
|
}
|
|
break;
|
|
case SWITCH_ABC_TYPE_CLOSE:
|
|
{
|
|
switch_codec_implementation_t read_impl = { 0 };
|
|
switch_core_session_get_read_impl(session, &read_impl);
|
|
|
|
switch_log_printf(SWITCH_CHANNEL_SESSION_LOG(session), SWITCH_LOG_DEBUG, "Stop recording file %s\n", rh->file);
|
|
switch_channel_set_private(channel, rh->file, NULL);
|
|
|
|
if (rh->native) {
|
|
switch_core_file_close(&rh->in_fh);
|
|
switch_core_file_close(&rh->out_fh);
|
|
} else if (rh->fh) {
|
|
switch_size_t len;
|
|
uint8_t data[SWITCH_RECOMMENDED_BUFFER_SIZE];
|
|
switch_frame_t frame = { 0 };
|
|
const char *file_trimmed_ms = NULL;
|
|
const char *file_size = NULL;
|
|
const char *file_trimmed = NULL;
|
|
|
|
if (rh->thread_ready) {
|
|
switch_status_t st;
|
|
|
|
rh->thread_ready = 0;
|
|
|
|
switch_mutex_lock(rh->cond_mutex);
|
|
switch_thread_cond_signal(rh->cond);
|
|
switch_mutex_unlock(rh->cond_mutex);
|
|
|
|
switch_thread_join(&st, rh->thread);
|
|
}
|
|
|
|
if (rh->thread_buffer) {
|
|
switch_buffer_destroy(&rh->thread_buffer);
|
|
}
|
|
|
|
frame.data = data;
|
|
frame.buflen = SWITCH_RECOMMENDED_BUFFER_SIZE;
|
|
|
|
while (switch_core_media_bug_read(bug, &frame, SWITCH_TRUE) == SWITCH_STATUS_SUCCESS) {
|
|
len = (switch_size_t) frame.datalen / 2;
|
|
|
|
if (len && switch_core_file_write(rh->fh, mask ? null_data : data, &len) != SWITCH_STATUS_SUCCESS) {
|
|
switch_log_printf(SWITCH_CHANNEL_SESSION_LOG(session), SWITCH_LOG_ERROR, "Error writing %s\n", rh->file);
|
|
/* File write failed */
|
|
set_completion_cause(rh, "uri-failure");
|
|
if (rh->hangup_on_error) {
|
|
switch_channel_hangup(channel, SWITCH_CAUSE_DESTINATION_OUT_OF_ORDER);
|
|
switch_core_session_reset(session, SWITCH_TRUE, SWITCH_TRUE);
|
|
}
|
|
send_record_stop_event(channel, &read_impl, rh);
|
|
record_helper_destroy(&rh, session);
|
|
|
|
return SWITCH_FALSE;
|
|
}
|
|
}
|
|
|
|
|
|
//if (switch_core_file_has_video(rh->fh, SWITCH_TRUE)) {
|
|
//switch_core_media_set_video_file(session, NULL, SWITCH_RW_READ);
|
|
//switch_channel_clear_flag_recursive(session->channel, CF_VIDEO_DECODED_READ);
|
|
//}
|
|
|
|
|
|
switch_core_file_pre_close(rh->fh);
|
|
switch_core_file_get_string(rh->fh, SWITCH_AUDIO_COL_STR_FILE_SIZE, &file_size);
|
|
switch_core_file_get_string(rh->fh, SWITCH_AUDIO_COL_STR_FILE_TRIMMED, &file_trimmed);
|
|
switch_core_file_get_string(rh->fh, SWITCH_AUDIO_COL_STR_FILE_TRIMMED_MS, &file_trimmed_ms);
|
|
if (file_trimmed_ms) switch_channel_set_variable(channel, "record_trimmed_ms", file_trimmed_ms);
|
|
if (file_size) switch_channel_set_variable(channel, "record_file_size", file_size);
|
|
if (file_trimmed) switch_channel_set_variable(channel, "record_trimmed", file_trimmed);
|
|
switch_core_file_close(rh->fh);
|
|
|
|
if (!rh->writes && !rh->vwrites && !switch_test_flag(rh->fh, SWITCH_FILE_WRITE_APPEND)) {
|
|
switch_log_printf(SWITCH_CHANNEL_SESSION_LOG(session), SWITCH_LOG_DEBUG, "Discarding empty file %s\n", rh->file);
|
|
switch_channel_set_variable(channel, "RECORD_DISCARDED", "true");
|
|
switch_file_remove(rh->file, switch_core_session_get_pool(session));
|
|
set_completion_cause(rh, "empty-file");
|
|
} else if ((rh->fh->samples_out < rh->fh->samplerate * rh->min_sec) && !switch_test_flag(rh->fh, SWITCH_FILE_WRITE_APPEND)) {
|
|
switch_log_printf(SWITCH_CHANNEL_SESSION_LOG(session), SWITCH_LOG_DEBUG, "Discarding short file %s\n", rh->file);
|
|
switch_channel_set_variable(channel, "RECORD_DISCARDED", "true");
|
|
switch_file_remove(rh->file, switch_core_session_get_pool(session));
|
|
set_completion_cause(rh, "input-too-short");
|
|
} else {
|
|
|
|
if (switch_channel_down_nosig(channel)) {
|
|
/* We got hung up */
|
|
switch_log_printf(SWITCH_CHANNEL_SESSION_LOG(session), SWITCH_LOG_DEBUG, "Channel is hung up\n");
|
|
if (rh->speech_detected) {
|
|
/* Treat it as equivalent with final-silence */
|
|
set_completion_cause(rh, "success-silence");
|
|
} else {
|
|
/* Treat it as equivalent with inital-silence timeout */
|
|
set_completion_cause(rh, "no-input-timeout");
|
|
}
|
|
} else {
|
|
/* Set the completion_cause to maxtime reached, unless it's already set */
|
|
set_completion_cause(rh, "success-maxtime");
|
|
}
|
|
}
|
|
}
|
|
|
|
send_record_stop_event(channel, &read_impl, rh);
|
|
record_helper_post_process(rh, session);
|
|
record_helper_destroy(&rh, session);
|
|
}
|
|
|
|
break;
|
|
case SWITCH_ABC_TYPE_READ_PING:
|
|
|
|
if (rh->fh) {
|
|
switch_size_t len;
|
|
uint8_t data[SWITCH_RECOMMENDED_BUFFER_SIZE];
|
|
switch_frame_t frame = { 0 };
|
|
switch_status_t status;
|
|
int i = 0;
|
|
|
|
frame.data = data;
|
|
frame.buflen = SWITCH_RECOMMENDED_BUFFER_SIZE;
|
|
|
|
for (;;) {
|
|
status = switch_core_media_bug_read(bug, &frame, i++ == 0 ? SWITCH_FALSE : SWITCH_TRUE);
|
|
|
|
if (status != SWITCH_STATUS_SUCCESS || !frame.datalen) {
|
|
break;
|
|
} else {
|
|
len = (switch_size_t) frame.datalen / 2 / frame.channels;
|
|
|
|
if (rh->thread_buffer) {
|
|
switch_mutex_lock(rh->buffer_mutex);
|
|
switch_buffer_write(rh->thread_buffer, mask ? null_data : data, frame.datalen);
|
|
switch_mutex_unlock(rh->buffer_mutex);
|
|
if (switch_mutex_trylock(rh->cond_mutex) == SWITCH_STATUS_SUCCESS) {
|
|
switch_thread_cond_signal(rh->cond);
|
|
switch_mutex_unlock(rh->cond_mutex);
|
|
}
|
|
} else if (switch_core_file_write(rh->fh, mask ? null_data : data, &len) != SWITCH_STATUS_SUCCESS) {
|
|
switch_log_printf(SWITCH_CHANNEL_SESSION_LOG(session), SWITCH_LOG_ERROR, "Error writing %s\n", rh->file);
|
|
/* File write failed */
|
|
set_completion_cause(rh, "uri-failure");
|
|
if (rh->hangup_on_error) {
|
|
switch_channel_hangup(channel, SWITCH_CAUSE_DESTINATION_OUT_OF_ORDER);
|
|
switch_core_session_reset(session, SWITCH_TRUE, SWITCH_TRUE);
|
|
}
|
|
return SWITCH_FALSE;
|
|
}
|
|
|
|
rh->writes++;
|
|
|
|
/* check for silence timeout */
|
|
if (rh->silence_threshold) {
|
|
switch_codec_implementation_t read_impl = { 0 };
|
|
switch_core_session_get_read_impl(session, &read_impl);
|
|
if (is_silence_frame(&frame, rh->silence_threshold, &read_impl)) {
|
|
if (!rh->silence_time) {
|
|
/* start of silence */
|
|
switch_log_printf(SWITCH_CHANNEL_SESSION_LOG(session), SWITCH_LOG_DEBUG, "Start of silence detected\n");
|
|
rh->silence_time = switch_micro_time_now();
|
|
} else {
|
|
/* continuing silence */
|
|
int duration_ms = (int)((switch_micro_time_now() - rh->silence_time) / 1000);
|
|
if (rh->silence_timeout_ms > 0 && duration_ms >= rh->silence_timeout_ms) {
|
|
switch_log_printf(SWITCH_CHANNEL_SESSION_LOG(session), SWITCH_LOG_DEBUG, "Recording file %s timeout: %i >= %i\n", rh->file, duration_ms, rh->silence_timeout_ms);
|
|
switch_core_media_bug_set_flag(bug, SMBF_PRUNE);
|
|
if (rh->speech_detected) {
|
|
/* Reached final silence timeout */
|
|
set_completion_cause(rh, "success-silence");
|
|
} else {
|
|
/* Reached initial silence timeout */
|
|
set_completion_cause(rh, "no-input-timeout");
|
|
/* Discard the silent file? */
|
|
}
|
|
}
|
|
}
|
|
} else { /* not silence */
|
|
if (rh->silence_time) {
|
|
switch_log_printf(SWITCH_CHANNEL_SESSION_LOG(session), SWITCH_LOG_DEBUG, "Start of speech detected\n");
|
|
rh->speech_detected = SWITCH_TRUE;
|
|
/* end of silence */
|
|
rh->silence_time = 0;
|
|
/* switch from initial timeout to final timeout */
|
|
rh->silence_timeout_ms = rh->final_timeout_ms;
|
|
}
|
|
}
|
|
} else {
|
|
/* no silence detection */
|
|
if (!rh->speech_detected) {
|
|
switch_log_printf(SWITCH_CHANNEL_SESSION_LOG(session), SWITCH_LOG_DEBUG, "No silence detection configured; assuming start of speech\n");
|
|
rh->speech_detected = SWITCH_TRUE;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
break;
|
|
case SWITCH_ABC_TYPE_READ_VIDEO_PING:
|
|
case SWITCH_ABC_TYPE_STREAM_VIDEO_PING:
|
|
if (rh->fh && switch_test_flag(rh->fh, SWITCH_FILE_OPEN)) {
|
|
if (!bug->video_ping_frame) break;
|
|
|
|
if (bug->video_ping_frame->img && switch_core_file_write_video(rh->fh, bug->video_ping_frame) != SWITCH_STATUS_SUCCESS &&
|
|
rh->hangup_on_error) {
|
|
switch_log_printf(SWITCH_CHANNEL_SESSION_LOG(session), SWITCH_LOG_ERROR, "Error writing video to %s\n", rh->file);
|
|
switch_channel_hangup(channel, SWITCH_CAUSE_DESTINATION_OUT_OF_ORDER);
|
|
switch_core_session_reset(session, SWITCH_TRUE, SWITCH_TRUE);
|
|
return SWITCH_FALSE;
|
|
}
|
|
rh->vwrites++;
|
|
}
|
|
break;
|
|
|
|
case SWITCH_ABC_TYPE_WRITE:
|
|
default:
|
|
break;
|
|
}
|
|
|
|
return SWITCH_TRUE;
|
|
}
|
|
|
|
|
|
static switch_bool_t text_callback(switch_media_bug_t *bug, void *user_data, switch_abc_type_t type)
|
|
{
|
|
|
|
switch (type) {
|
|
case SWITCH_ABC_TYPE_READ_TEXT:
|
|
{
|
|
const char *text = switch_core_media_bug_get_text(bug);
|
|
|
|
|
|
if (!zstr(text)) {
|
|
switch_event_t *event = NULL;
|
|
switch_core_session_t *session = switch_core_media_bug_get_session(bug);
|
|
//switch_channel_t *channel = switch_core_session_get_channel(session);
|
|
|
|
if (switch_event_create(&event, SWITCH_EVENT_TEXT) == SWITCH_STATUS_SUCCESS) {
|
|
switch_event_add_body(event, text, SWITCH_VA_NONE);
|
|
|
|
if (switch_true(switch_core_get_variable("fire_text_events"))) {
|
|
switch_event_t *clone = NULL;
|
|
|
|
switch_event_dup(&clone, event);
|
|
switch_event_fire(&clone);
|
|
}
|
|
|
|
switch_core_session_queue_event(session, &event);
|
|
}
|
|
}
|
|
}
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
|
|
return SWITCH_TRUE;
|
|
}
|
|
|
|
SWITCH_DECLARE(switch_status_t) switch_ivr_capture_text(switch_core_session_t *session, switch_bool_t on)
|
|
{
|
|
switch_media_bug_t *bug;
|
|
switch_channel_t *channel = switch_core_session_get_channel(session);
|
|
|
|
bug = (switch_media_bug_t *) switch_channel_get_private(channel, "capture_text");
|
|
|
|
if (on) {
|
|
|
|
if (bug) {
|
|
switch_log_printf(SWITCH_CHANNEL_SESSION_LOG(session), SWITCH_LOG_ERROR, "text bug already attached\n");
|
|
return SWITCH_STATUS_FALSE;
|
|
}
|
|
|
|
|
|
if (switch_core_media_bug_add(session, "capture_text", switch_core_session_get_uuid(session),
|
|
text_callback, NULL, 0,
|
|
SMBF_READ_TEXT_STREAM,
|
|
&bug) != SWITCH_STATUS_SUCCESS) {
|
|
switch_log_printf(SWITCH_CHANNEL_SESSION_LOG(session), SWITCH_LOG_ERROR, "Cannot attach bug\n");
|
|
return SWITCH_STATUS_FALSE;
|
|
}
|
|
|
|
switch_channel_set_private(channel, "capture_text", bug);
|
|
return SWITCH_STATUS_SUCCESS;
|
|
|
|
} else {
|
|
|
|
if (bug) {
|
|
switch_channel_set_private(channel, "capture_text", NULL);
|
|
switch_core_media_bug_remove(session, &bug);
|
|
return SWITCH_STATUS_SUCCESS;
|
|
} else {
|
|
switch_log_printf(SWITCH_CHANNEL_SESSION_LOG(session), SWITCH_LOG_ERROR, "text bug not attached\n");
|
|
return SWITCH_STATUS_FALSE;
|
|
}
|
|
|
|
}
|
|
}
|
|
|
|
|
|
SWITCH_DECLARE(switch_status_t) switch_ivr_record_session_mask(switch_core_session_t *session, const char *file, switch_bool_t on)
|
|
{
|
|
switch_media_bug_t *bug;
|
|
switch_channel_t *channel = switch_core_session_get_channel(session);
|
|
|
|
if ((bug = switch_channel_get_private(channel, file))) {
|
|
if (on) {
|
|
switch_core_media_bug_set_flag(bug, SMBF_MASK);
|
|
} else {
|
|
switch_core_media_bug_clear_flag(bug, SMBF_MASK);
|
|
}
|
|
return SWITCH_STATUS_SUCCESS;
|
|
}
|
|
return SWITCH_STATUS_FALSE;
|
|
}
|
|
|
|
SWITCH_DECLARE(switch_status_t) switch_ivr_record_session_pause(switch_core_session_t *session, const char *file, switch_bool_t on)
|
|
{
|
|
switch_media_bug_t *bug;
|
|
switch_channel_t *channel = switch_core_session_get_channel(session);
|
|
|
|
if ((bug = switch_channel_get_private(channel, file))) {
|
|
if (on) {
|
|
switch_core_media_bug_set_flag(bug, SMBF_PAUSE);
|
|
} else {
|
|
switch_core_media_bug_clear_flag(bug, SMBF_PAUSE);
|
|
}
|
|
return SWITCH_STATUS_SUCCESS;
|
|
}
|
|
return SWITCH_STATUS_FALSE;
|
|
}
|
|
|
|
SWITCH_DECLARE(switch_status_t) switch_ivr_stop_record_session(switch_core_session_t *session, const char *file)
|
|
{
|
|
switch_media_bug_t *bug;
|
|
switch_channel_t *channel = switch_core_session_get_channel(session);
|
|
|
|
if (!strcasecmp(file, "all")) {
|
|
return switch_core_media_bug_remove_callback(session, record_callback);
|
|
} else if ((bug = switch_channel_get_private(channel, file))) {
|
|
switch_core_media_bug_remove(session, &bug);
|
|
return SWITCH_STATUS_SUCCESS;
|
|
}
|
|
return SWITCH_STATUS_FALSE;
|
|
}
|
|
|
|
static void* switch_ivr_record_user_data_dup(switch_core_session_t *session, void *user_data)
|
|
{
|
|
struct record_helper *rh = (struct record_helper *) user_data;
|
|
|
|
if (!rh->transfer_complete && session == rh->transfer_from_session) {
|
|
/* Transfer failed and now we are called to put the original session back */
|
|
rh->recording_session = rh->transfer_from_session;
|
|
rh->transfer_from_session = NULL;
|
|
} else {
|
|
rh->transfer_from_session = rh->recording_session;
|
|
rh->recording_session = session;
|
|
rh->transfer_complete = 0;
|
|
}
|
|
|
|
return rh;
|
|
}
|
|
|
|
SWITCH_DECLARE(switch_status_t) switch_ivr_transfer_recordings(switch_core_session_t *orig_session, switch_core_session_t *new_session)
|
|
{
|
|
const char *var = NULL;
|
|
switch_channel_t *orig_channel = switch_core_session_get_channel(orig_session);
|
|
switch_channel_t *new_channel = switch_core_session_get_channel(new_session);
|
|
|
|
if ((var = switch_channel_get_variable(orig_channel, SWITCH_RECORD_POST_PROCESS_EXEC_API_VARIABLE))) {
|
|
switch_channel_set_variable(new_channel, SWITCH_RECORD_POST_PROCESS_EXEC_API_VARIABLE, var);
|
|
}
|
|
switch_channel_transfer_variable_prefix(orig_channel, new_channel, SWITCH_RECORD_POST_PROCESS_EXEC_APP_VARIABLE);
|
|
|
|
return switch_core_media_bug_transfer_callback(orig_session, new_session, record_callback, switch_ivr_record_user_data_dup);
|
|
}
|
|
|
|
struct eavesdrop_pvt {
|
|
switch_buffer_t *buffer;
|
|
switch_mutex_t *mutex;
|
|
switch_buffer_t *r_buffer;
|
|
switch_mutex_t *r_mutex;
|
|
switch_buffer_t *w_buffer;
|
|
switch_mutex_t *w_mutex;
|
|
switch_core_session_t *eavesdropper;
|
|
uint32_t flags;
|
|
switch_frame_t demux_frame;
|
|
int set_decoded_read;
|
|
int errs;
|
|
switch_codec_implementation_t read_impl;
|
|
switch_codec_implementation_t tread_impl;
|
|
uint8_t data[SWITCH_RECOMMENDED_BUFFER_SIZE];
|
|
};
|
|
|
|
|
|
static switch_status_t video_eavesdrop_callback(switch_core_session_t *session, switch_frame_t *frame, void *user_data)
|
|
{
|
|
switch_media_bug_t *bug = (switch_media_bug_t *) user_data;
|
|
|
|
if (frame->img) {
|
|
if (switch_core_media_bug_test_flag(bug, SMBF_SPY_VIDEO_STREAM)) {
|
|
switch_core_media_bug_push_spy_frame(bug, frame, SWITCH_RW_READ);
|
|
}
|
|
|
|
if (switch_core_media_bug_test_flag(bug, SMBF_SPY_VIDEO_STREAM_BLEG)) {
|
|
switch_core_media_bug_push_spy_frame(bug, frame, SWITCH_RW_WRITE);
|
|
}
|
|
}
|
|
|
|
return SWITCH_STATUS_SUCCESS;
|
|
}
|
|
|
|
static switch_bool_t eavesdrop_callback(switch_media_bug_t *bug, void *user_data, switch_abc_type_t type)
|
|
{
|
|
struct eavesdrop_pvt *ep = (struct eavesdrop_pvt *) user_data;
|
|
uint8_t data[SWITCH_RECOMMENDED_BUFFER_SIZE];
|
|
switch_frame_t frame = { 0 };
|
|
switch_core_session_t *session = switch_core_media_bug_get_session(bug);
|
|
switch_channel_t *e_channel = switch_core_session_get_channel(ep->eavesdropper);
|
|
int show_spy = 0;
|
|
switch_frame_t *nframe = NULL;
|
|
|
|
frame.data = data;
|
|
frame.buflen = SWITCH_RECOMMENDED_BUFFER_SIZE;
|
|
|
|
show_spy = switch_core_media_bug_test_flag(bug, SMBF_SPY_VIDEO_STREAM) || switch_core_media_bug_test_flag(bug, SMBF_SPY_VIDEO_STREAM_BLEG);
|
|
|
|
if (show_spy) {
|
|
if (!ep->set_decoded_read) {
|
|
ep->set_decoded_read = 1;
|
|
switch_channel_set_flag_recursive(e_channel, CF_VIDEO_DECODED_READ);
|
|
switch_core_session_request_video_refresh(ep->eavesdropper);
|
|
}
|
|
} else {
|
|
if (ep->set_decoded_read) {
|
|
ep->set_decoded_read = 0;
|
|
switch_channel_clear_flag_recursive(e_channel, CF_VIDEO_DECODED_READ);
|
|
switch_core_session_request_video_refresh(ep->eavesdropper);
|
|
}
|
|
}
|
|
|
|
switch (type) {
|
|
case SWITCH_ABC_TYPE_INIT:
|
|
|
|
if (switch_core_media_bug_test_flag(bug, SMBF_READ_VIDEO_STREAM) ||
|
|
switch_core_media_bug_test_flag(bug, SMBF_WRITE_VIDEO_STREAM) ||
|
|
switch_core_media_bug_test_flag(bug, SMBF_READ_VIDEO_PING)) {
|
|
switch_core_session_set_video_read_callback(ep->eavesdropper, video_eavesdrop_callback, (void *)bug);
|
|
switch_channel_set_flag_recursive(switch_core_session_get_channel(session), CF_VIDEO_DECODED_READ);
|
|
}
|
|
break;
|
|
case SWITCH_ABC_TYPE_CLOSE:
|
|
if (ep->set_decoded_read) {
|
|
switch_channel_clear_flag_recursive(e_channel, CF_VIDEO_DECODED_READ);
|
|
}
|
|
|
|
if (switch_core_media_bug_test_flag(bug, SMBF_READ_VIDEO_STREAM) ||
|
|
switch_core_media_bug_test_flag(bug, SMBF_WRITE_VIDEO_STREAM) ||
|
|
switch_core_media_bug_test_flag(bug, SMBF_READ_VIDEO_PING)) {
|
|
switch_core_session_set_video_read_callback(ep->eavesdropper, NULL, NULL);
|
|
}
|
|
|
|
switch_channel_clear_flag_recursive(switch_core_session_get_channel(session), CF_VIDEO_DECODED_READ);
|
|
|
|
break;
|
|
case SWITCH_ABC_TYPE_TAP_NATIVE_WRITE:
|
|
nframe = switch_core_media_bug_get_native_write_frame(bug);
|
|
break;
|
|
case SWITCH_ABC_TYPE_TAP_NATIVE_READ:
|
|
nframe = switch_core_media_bug_get_native_read_frame(bug);
|
|
break;
|
|
case SWITCH_ABC_TYPE_WRITE:
|
|
break;
|
|
case SWITCH_ABC_TYPE_READ_PING:
|
|
if (ep->buffer) {
|
|
if (switch_core_media_bug_read(bug, &frame, SWITCH_FALSE) != SWITCH_STATUS_FALSE) {
|
|
switch_buffer_lock(ep->buffer);
|
|
switch_buffer_zwrite(ep->buffer, frame.data, frame.datalen);
|
|
switch_buffer_unlock(ep->buffer);
|
|
}
|
|
}
|
|
break;
|
|
case SWITCH_ABC_TYPE_READ:
|
|
break;
|
|
|
|
case SWITCH_ABC_TYPE_READ_REPLACE:
|
|
{
|
|
|
|
if (switch_test_flag(ep, ED_MUX_READ)) {
|
|
switch_frame_t *rframe = switch_core_media_bug_get_read_replace_frame(bug);
|
|
|
|
if (switch_buffer_inuse(ep->r_buffer) >= rframe->datalen) {
|
|
uint32_t bytes;
|
|
int channels = rframe->channels ? rframe->channels : 1;
|
|
|
|
switch_buffer_lock(ep->r_buffer);
|
|
bytes = (uint32_t) switch_buffer_read(ep->r_buffer, ep->data, rframe->datalen);
|
|
|
|
rframe->datalen = switch_merge_sln(rframe->data, rframe->samples, (int16_t *) ep->data, bytes / 2, channels) * 2 * channels;
|
|
rframe->samples = rframe->datalen / 2;
|
|
|
|
ep->demux_frame.data = ep->data;
|
|
ep->demux_frame.datalen = bytes;
|
|
ep->demux_frame.samples = bytes / 2;
|
|
ep->demux_frame.channels = rframe->channels;
|
|
|
|
switch_buffer_unlock(ep->r_buffer);
|
|
switch_core_media_bug_set_read_replace_frame(bug, rframe);
|
|
switch_core_media_bug_set_read_demux_frame(bug, &ep->demux_frame);
|
|
}
|
|
}
|
|
|
|
}
|
|
break;
|
|
|
|
case SWITCH_ABC_TYPE_WRITE_REPLACE:
|
|
{
|
|
if (switch_test_flag(ep, ED_MUX_WRITE)) {
|
|
switch_frame_t *rframe = switch_core_media_bug_get_write_replace_frame(bug);
|
|
|
|
if (switch_buffer_inuse(ep->w_buffer) >= rframe->datalen) {
|
|
uint32_t bytes;
|
|
int channels = rframe->channels ? rframe->channels : 1;
|
|
|
|
switch_buffer_lock(ep->w_buffer);
|
|
bytes = (uint32_t) switch_buffer_read(ep->w_buffer, data, rframe->datalen);
|
|
|
|
rframe->datalen = switch_merge_sln(rframe->data, rframe->samples, (int16_t *) data, bytes / 2, channels) * 2 * channels;
|
|
rframe->samples = rframe->datalen / 2;
|
|
|
|
switch_buffer_unlock(ep->w_buffer);
|
|
switch_core_media_bug_set_write_replace_frame(bug, rframe);
|
|
}
|
|
}
|
|
}
|
|
break;
|
|
|
|
case SWITCH_ABC_TYPE_READ_VIDEO_PING:
|
|
case SWITCH_ABC_TYPE_STREAM_VIDEO_PING:
|
|
{
|
|
|
|
if (!bug->video_ping_frame || !bug->video_ping_frame->img) {
|
|
break;
|
|
}
|
|
|
|
if (switch_core_session_read_lock(ep->eavesdropper) == SWITCH_STATUS_SUCCESS) {
|
|
if (switch_core_session_write_video_frame(ep->eavesdropper, bug->video_ping_frame, SWITCH_IO_FLAG_NONE, 0) != SWITCH_STATUS_SUCCESS) {
|
|
switch_log_printf(SWITCH_CHANNEL_SESSION_LOG(session), SWITCH_LOG_ERROR, "Error writing video to %s\n", switch_core_session_get_name(ep->eavesdropper));
|
|
ep->errs++;
|
|
|
|
if (ep->errs > 10) {
|
|
switch_channel_hangup(e_channel, SWITCH_CAUSE_DESTINATION_OUT_OF_ORDER);
|
|
switch_core_session_reset(ep->eavesdropper, SWITCH_TRUE, SWITCH_TRUE);
|
|
switch_core_session_rwunlock(ep->eavesdropper);
|
|
return SWITCH_FALSE;
|
|
}
|
|
} else {
|
|
ep->errs = 0;
|
|
}
|
|
switch_core_session_rwunlock(ep->eavesdropper);
|
|
}
|
|
}
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
|
|
if (nframe) {
|
|
switch_frame_t frame = *nframe;
|
|
uint8_t buf[SWITCH_RECOMMENDED_BUFFER_SIZE] = "";
|
|
|
|
frame.data = buf;
|
|
frame.codec = nframe->codec;
|
|
|
|
memcpy(frame.data, nframe->data, nframe->datalen);
|
|
|
|
if (switch_core_session_write_frame(ep->eavesdropper, &frame, SWITCH_IO_FLAG_NONE, 0) != SWITCH_STATUS_SUCCESS) {
|
|
return SWITCH_FALSE;
|
|
}
|
|
}
|
|
|
|
return SWITCH_TRUE;
|
|
}
|
|
|
|
SWITCH_DECLARE(switch_status_t) switch_ivr_eavesdrop_pop_eavesdropper(switch_core_session_t *session, switch_core_session_t **sessionp)
|
|
{
|
|
switch_media_bug_t *bug;
|
|
switch_status_t status = SWITCH_STATUS_FALSE;
|
|
|
|
if (switch_core_media_bug_pop(session, "eavesdrop", &bug) == SWITCH_STATUS_SUCCESS) {
|
|
struct eavesdrop_pvt *ep = (struct eavesdrop_pvt *) switch_core_media_bug_get_user_data(bug);
|
|
|
|
if (ep && ep->eavesdropper && ep->eavesdropper != session) {
|
|
if (switch_core_session_read_lock(ep->eavesdropper) == SWITCH_STATUS_SUCCESS) {
|
|
*sessionp = ep->eavesdropper;
|
|
switch_core_media_bug_set_flag(bug, SMBF_PRUNE);
|
|
status = SWITCH_STATUS_SUCCESS;
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
return status;
|
|
}
|
|
|
|
struct exec_cb_data {
|
|
switch_core_session_t *caller;
|
|
char *var;
|
|
char *val;
|
|
};
|
|
|
|
static void exec_cb(switch_media_bug_t *bug, void *user_data)
|
|
{
|
|
struct exec_cb_data *data = (struct exec_cb_data *) user_data;
|
|
struct eavesdrop_pvt *ep = (struct eavesdrop_pvt *) switch_core_media_bug_get_user_data(bug);
|
|
|
|
if (ep && ep->eavesdropper && ep->eavesdropper != data->caller) {
|
|
switch_channel_t *a = switch_core_session_get_channel(ep->eavesdropper);
|
|
switch_channel_t *b = switch_core_session_get_channel(data->caller);
|
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_DEBUG, "%s telling %s to exec %s:%s\n",
|
|
switch_channel_get_name(b), switch_channel_get_name(a), data->var, data->val);
|
|
|
|
switch_core_session_execute_application(ep->eavesdropper, data->var, data->val);
|
|
}
|
|
}
|
|
|
|
static void display_exec_cb(switch_media_bug_t *bug, void *user_data)
|
|
{
|
|
struct exec_cb_data *data = (struct exec_cb_data *) user_data;
|
|
struct eavesdrop_pvt *ep = (struct eavesdrop_pvt *) switch_core_media_bug_get_user_data(bug);
|
|
|
|
if (ep && ep->eavesdropper && ep->eavesdropper != data->caller) {
|
|
switch_core_session_message_t msg = { 0 };
|
|
|
|
msg.from = __FILE__;
|
|
msg.message_id = SWITCH_MESSAGE_INDICATE_DISPLAY;
|
|
msg.string_array_arg[0] = data->var;
|
|
msg.string_array_arg[1] = data->val;
|
|
|
|
switch_core_session_receive_message(ep->eavesdropper, &msg);
|
|
}
|
|
}
|
|
|
|
SWITCH_DECLARE(switch_status_t) switch_ivr_eavesdrop_exec_all(switch_core_session_t *session, const char *app, const char *arg)
|
|
{
|
|
struct exec_cb_data *data = NULL;
|
|
|
|
data = switch_core_session_alloc(session, sizeof(*data));
|
|
data->var = switch_core_session_strdup(session, app);
|
|
data->val = switch_core_session_strdup(session, arg);
|
|
data->caller = session;
|
|
|
|
return switch_core_media_bug_exec_all(session, "eavesdrop", exec_cb, data);
|
|
}
|
|
|
|
|
|
SWITCH_DECLARE(switch_status_t) switch_ivr_eavesdrop_update_display(switch_core_session_t *session, const char *name, const char *number)
|
|
{
|
|
struct exec_cb_data *data = NULL;
|
|
switch_channel_t *channel = switch_core_session_get_channel(session);
|
|
switch_status_t status = SWITCH_STATUS_FALSE;
|
|
|
|
data = switch_core_session_alloc(session, sizeof(*data));
|
|
data->var = switch_core_session_strdup(session, name);
|
|
data->val = switch_core_session_strdup(session, number);
|
|
data->caller = session;
|
|
|
|
if (!switch_channel_test_app_flag_key("EAVESDROP", channel, 1)) {
|
|
switch_channel_set_app_flag_key("EAVESDROP", channel, 1);
|
|
status = switch_core_media_bug_exec_all(session, "eavesdrop", display_exec_cb, data);
|
|
switch_channel_clear_app_flag_key("EAVESDROP", channel, 1);
|
|
}
|
|
|
|
return status;
|
|
}
|
|
|
|
/*Greatest Common Divisor*/
|
|
static uint32_t switch_gcd(uint32_t x, uint32_t y)
|
|
{
|
|
if (y == 0) {
|
|
return x;
|
|
}
|
|
|
|
return switch_gcd(y, x % y);
|
|
}
|
|
|
|
/*Least Common Multiple*/
|
|
static uint32_t switch_lcm(uint32_t x, uint32_t y)
|
|
{
|
|
uint32_t gcd = switch_gcd(x, y);
|
|
|
|
if (gcd) return (x * y) / gcd;
|
|
|
|
return 0;
|
|
}
|
|
|
|
SWITCH_DECLARE(switch_status_t) switch_ivr_eavesdrop_session(switch_core_session_t *session,
|
|
const char *uuid, const char *require_group, switch_eavesdrop_flag_t flags)
|
|
{
|
|
switch_core_session_t *tsession;
|
|
switch_status_t status = SWITCH_STATUS_FALSE;
|
|
switch_channel_t *channel = switch_core_session_get_channel(session);
|
|
int codec_initialized = 0;
|
|
const char *name, *num;
|
|
|
|
if ((tsession = switch_core_session_locate(uuid))) {
|
|
struct eavesdrop_pvt *ep = NULL;
|
|
switch_media_bug_t *bug = NULL;
|
|
switch_channel_t *tchannel = switch_core_session_get_channel(tsession);
|
|
switch_frame_t *read_frame, write_frame = { 0 };
|
|
switch_codec_t codec = { 0 };
|
|
int16_t buf[SWITCH_RECOMMENDED_BUFFER_SIZE / 2];
|
|
uint32_t tlen;
|
|
const char *macro_name = "eavesdrop_announce";
|
|
const char *id_name = NULL;
|
|
switch_codec_implementation_t tread_impl = { 0 }, read_impl = { 0 };
|
|
switch_core_session_message_t msg = { 0 };
|
|
char cid_buf[1024] = "";
|
|
switch_caller_profile_t *cp = NULL;
|
|
uint32_t sanity = 600;
|
|
switch_media_bug_flag_t read_flags = 0, write_flags = 0, stereo_flag = 0;
|
|
const char *vval;
|
|
int buf_size = 0;
|
|
int channels;
|
|
int lcm, buff_min_len, buffered = 1;
|
|
|
|
if (!switch_channel_media_up(channel)) {
|
|
goto end;
|
|
}
|
|
|
|
if (tsession == session) {
|
|
switch_log_printf(SWITCH_CHANNEL_SESSION_LOG(session), SWITCH_LOG_ERROR, "Eavesdrop target invalid.\n");
|
|
goto end;
|
|
}
|
|
|
|
while(switch_channel_state_change_pending(tchannel) || !switch_channel_media_up(tchannel)) {
|
|
switch_yield(10000);
|
|
if (!--sanity) break;
|
|
}
|
|
|
|
if (!switch_channel_media_up(tchannel)) {
|
|
goto end;
|
|
}
|
|
|
|
if ((flags & ED_TAP_READ) || (flags & ED_TAP_WRITE)) {
|
|
switch_core_session_get_real_read_impl(tsession, &tread_impl);
|
|
switch_core_session_get_real_read_impl(session, &read_impl);
|
|
|
|
if (strcasecmp(tread_impl.iananame, read_impl.iananame)) {
|
|
switch_log_printf(SWITCH_CHANNEL_SESSION_LOG(session), SWITCH_LOG_ERROR, "Codecs do not match which is required for this mode\n");
|
|
goto end;
|
|
}
|
|
}
|
|
|
|
switch_core_session_get_read_impl(tsession, &tread_impl);
|
|
switch_core_session_get_read_impl(session, &read_impl);
|
|
|
|
if ((id_name = switch_channel_get_variable(tchannel, "eavesdrop_announce_id"))) {
|
|
const char *tmp = switch_channel_get_variable(tchannel, "eavesdrop_announce_macro");
|
|
if (tmp) {
|
|
macro_name = tmp;
|
|
}
|
|
|
|
switch_ivr_phrase_macro(session, macro_name, id_name, NULL, NULL);
|
|
}
|
|
|
|
|
|
if (!zstr(require_group)) {
|
|
int argc, i;
|
|
int ok = 0;
|
|
char *argv[10] = { 0 };
|
|
char *data;
|
|
|
|
const char *group_name = switch_channel_get_variable(tchannel, "eavesdrop_group");
|
|
/* If we don't have a group, then return */
|
|
if (!group_name) {
|
|
switch_goto_status(SWITCH_STATUS_BREAK, end);
|
|
}
|
|
/* Separate the group */
|
|
data = strdup(group_name);
|
|
if ((argc = switch_separate_string(data, ',', argv, (sizeof(argv) / sizeof(argv[0]))))) {
|
|
for (i = 0; i < argc; i++) {
|
|
/* If one of the group matches, then ok */
|
|
if (argv[i] && !strcmp(argv[i], require_group)) {
|
|
ok = 1;
|
|
}
|
|
}
|
|
}
|
|
switch_safe_free(data);
|
|
/* If we didn't find any match, then end */
|
|
if (!ok) {
|
|
switch_goto_status(SWITCH_STATUS_BREAK, end);
|
|
}
|
|
}
|
|
|
|
|
|
ep = switch_core_session_alloc(session, sizeof(*ep));
|
|
|
|
|
|
if (switch_channel_pre_answer(channel) != SWITCH_STATUS_SUCCESS) {
|
|
goto end;
|
|
}
|
|
|
|
if (tread_impl.decoded_bytes_per_packet < read_impl.decoded_bytes_per_packet) {
|
|
if (switch_core_codec_init(&codec,
|
|
"L16",
|
|
NULL,
|
|
NULL,
|
|
read_impl.actual_samples_per_second,
|
|
read_impl.microseconds_per_packet / 1000,
|
|
read_impl.number_of_channels,
|
|
SWITCH_CODEC_FLAG_ENCODE | SWITCH_CODEC_FLAG_DECODE,
|
|
NULL, switch_core_session_get_pool(session)) != SWITCH_STATUS_SUCCESS) {
|
|
switch_log_printf(SWITCH_CHANNEL_SESSION_LOG(session), SWITCH_LOG_ERROR, "Cannot init codec\n");
|
|
goto end;
|
|
}
|
|
} else {
|
|
if (switch_core_codec_init(&codec,
|
|
"L16",
|
|
NULL,
|
|
NULL,
|
|
tread_impl.actual_samples_per_second,
|
|
tread_impl.microseconds_per_packet / 1000,
|
|
tread_impl.number_of_channels,
|
|
SWITCH_CODEC_FLAG_ENCODE | SWITCH_CODEC_FLAG_DECODE,
|
|
NULL, switch_core_session_get_pool(session)) != SWITCH_STATUS_SUCCESS) {
|
|
switch_log_printf(SWITCH_CHANNEL_SESSION_LOG(session), SWITCH_LOG_ERROR, "Cannot init codec\n");
|
|
goto end;
|
|
}
|
|
buffered = 0;
|
|
}
|
|
|
|
ep->read_impl = read_impl;
|
|
ep->tread_impl = tread_impl;
|
|
|
|
codec_initialized = 1;
|
|
|
|
switch_core_session_set_read_codec(session, &codec);
|
|
write_frame.codec = &codec;
|
|
write_frame.data = buf;
|
|
write_frame.buflen = sizeof(buf);
|
|
write_frame.rate = codec.implementation->actual_samples_per_second;
|
|
|
|
/* Make sure that at least one leg is bridged, default to both */
|
|
if (! (flags & (ED_BRIDGE_READ | ED_BRIDGE_WRITE))) {
|
|
flags |= ED_BRIDGE_READ | ED_BRIDGE_WRITE;
|
|
}
|
|
|
|
buf_size = codec.implementation->decoded_bytes_per_packet * 10;
|
|
|
|
ep->eavesdropper = session;
|
|
ep->flags = flags;
|
|
|
|
if (!(flags & ED_TAP_READ) && !(flags & ED_TAP_WRITE)) {
|
|
switch_mutex_init(&ep->mutex, SWITCH_MUTEX_NESTED, switch_core_session_get_pool(tsession));
|
|
switch_buffer_create_dynamic(&ep->buffer, buf_size, buf_size, buf_size);
|
|
switch_buffer_add_mutex(ep->buffer, ep->mutex);
|
|
|
|
switch_mutex_init(&ep->w_mutex, SWITCH_MUTEX_NESTED, switch_core_session_get_pool(tsession));
|
|
switch_buffer_create_dynamic(&ep->w_buffer, buf_size, buf_size, buf_size);
|
|
switch_buffer_add_mutex(ep->w_buffer, ep->w_mutex);
|
|
|
|
switch_mutex_init(&ep->r_mutex, SWITCH_MUTEX_NESTED, switch_core_session_get_pool(tsession));
|
|
switch_buffer_create_dynamic(&ep->r_buffer, buf_size, buf_size, buf_size);
|
|
switch_buffer_add_mutex(ep->r_buffer, ep->r_mutex);
|
|
}
|
|
|
|
if (flags & ED_BRIDGE_READ) {
|
|
read_flags = SMBF_READ_STREAM | SMBF_READ_REPLACE;
|
|
}
|
|
|
|
if (flags & ED_BRIDGE_WRITE) {
|
|
write_flags = SMBF_WRITE_STREAM | SMBF_WRITE_REPLACE;
|
|
}
|
|
|
|
if (flags & ED_TAP_READ) {
|
|
read_flags = SMBF_TAP_NATIVE_READ;
|
|
write_flags = 0;
|
|
}
|
|
|
|
if (flags & ED_TAP_WRITE) {
|
|
write_flags = SMBF_TAP_NATIVE_WRITE;
|
|
read_flags = 0;
|
|
}
|
|
|
|
if (flags & ED_STEREO) {
|
|
stereo_flag = SMBF_STEREO;
|
|
}
|
|
|
|
if (switch_channel_test_flag(session->channel, CF_VIDEO) && switch_channel_test_flag(tsession->channel, CF_VIDEO)) {
|
|
if ((vval = switch_channel_get_variable(session->channel, "eavesdrop_show_listener_video"))) {
|
|
if (switch_true(vval) || !strcasecmp(vval, "aleg") || !strcasecmp(vval, "bleg") || !strcasecmp(vval, "both")) {
|
|
read_flags |= SMBF_SPY_VIDEO_STREAM;
|
|
}
|
|
if (switch_true(vval) || !strcasecmp(vval, "bleg") || !strcasecmp(vval, "both")) {
|
|
read_flags |= SMBF_SPY_VIDEO_STREAM_BLEG;
|
|
}
|
|
}
|
|
|
|
if ((vval = switch_channel_get_variable(session->channel, "eavesdrop_concat_video")) && switch_true(vval)) {
|
|
read_flags |= SMBF_READ_VIDEO_STREAM;
|
|
read_flags |= SMBF_WRITE_VIDEO_STREAM;
|
|
} else {
|
|
read_flags |= SMBF_READ_VIDEO_PING;
|
|
}
|
|
} else {
|
|
read_flags &= ~SMBF_READ_VIDEO_PING;
|
|
read_flags &= ~SMBF_READ_VIDEO_STREAM;
|
|
read_flags &= ~SMBF_WRITE_VIDEO_STREAM;
|
|
read_flags &= ~SMBF_SPY_VIDEO_STREAM;
|
|
read_flags &= ~SMBF_SPY_VIDEO_STREAM_BLEG;
|
|
}
|
|
|
|
|
|
if (switch_core_media_bug_add(tsession, "eavesdrop", uuid,
|
|
eavesdrop_callback, ep, 0,
|
|
read_flags | write_flags | SMBF_READ_PING | SMBF_THREAD_LOCK | SMBF_NO_PAUSE | stereo_flag,
|
|
&bug) != SWITCH_STATUS_SUCCESS) {
|
|
switch_log_printf(SWITCH_CHANNEL_SESSION_LOG(session), SWITCH_LOG_ERROR, "Cannot attach bug\n");
|
|
goto end;
|
|
}
|
|
|
|
if ((vval = switch_channel_get_variable(session->channel, "eavesdrop_video_spy_fmt"))) {
|
|
switch_media_bug_set_spy_fmt(bug, switch_media_bug_parse_spy_fmt(vval));
|
|
}
|
|
|
|
msg.from = __FILE__;
|
|
|
|
/* Tell the channel we are going to be in a bridge */
|
|
msg.message_id = SWITCH_MESSAGE_INDICATE_BRIDGE;
|
|
switch_core_session_receive_message(session, &msg);
|
|
cp = switch_channel_get_caller_profile(tchannel);
|
|
|
|
name = cp->caller_id_name;
|
|
num = cp->caller_id_number;
|
|
|
|
if ((flags & ED_TAP_READ) || (flags & ED_TAP_WRITE)) {
|
|
flags &= ~ED_DTMF;
|
|
flags &= ~ED_BRIDGE_READ;
|
|
flags &= ~ED_BRIDGE_WRITE;
|
|
}
|
|
|
|
if (flags & ED_COPY_DISPLAY) {
|
|
if (switch_channel_test_flag(tchannel, CF_BRIDGE_ORIGINATOR) || !switch_channel_test_flag(tchannel, CF_BRIDGED)) {
|
|
name = cp->callee_id_name;
|
|
num = cp->callee_id_number;
|
|
} else {
|
|
name = cp->caller_id_name;
|
|
num = cp->caller_id_number;
|
|
}
|
|
}
|
|
|
|
sanity = 300;
|
|
while(switch_channel_up(channel) && !switch_channel_media_ack(channel) && --sanity) {
|
|
switch_yield(10000);
|
|
}
|
|
|
|
|
|
switch_snprintf(cid_buf, sizeof(cid_buf), "%s|%s", name, num);
|
|
msg.string_arg = cid_buf;
|
|
msg.message_id = SWITCH_MESSAGE_INDICATE_DISPLAY;
|
|
switch_core_session_receive_message(session, &msg);
|
|
|
|
if (switch_channel_test_flag(tchannel, CF_VIDEO)) {
|
|
|
|
msg.message_id = SWITCH_MESSAGE_INDICATE_VIDEO_REFRESH_REQ;
|
|
|
|
switch_core_session_receive_message(tsession, &msg);
|
|
}
|
|
|
|
lcm = switch_lcm(tread_impl.decoded_bytes_per_packet, read_impl.decoded_bytes_per_packet);
|
|
|
|
while (switch_channel_up_nosig(tchannel) && switch_channel_ready(channel)) {
|
|
uint32_t len = sizeof(buf);
|
|
switch_event_t *event = NULL;
|
|
char *fcommand = NULL;
|
|
char db[2] = "";
|
|
int vid_bug = 0, vid_dual = 0;
|
|
|
|
status = switch_core_session_read_frame(session, &read_frame, SWITCH_IO_FLAG_NONE, 0);
|
|
|
|
if (!SWITCH_READ_ACCEPTABLE(status)) {
|
|
goto end_loop;
|
|
}
|
|
|
|
if (switch_core_session_dequeue_event(session, &event, SWITCH_FALSE) == SWITCH_STATUS_SUCCESS) {
|
|
char *command = switch_event_get_header(event, "eavesdrop-command");
|
|
if (command) {
|
|
fcommand = switch_core_session_strdup(session, command);
|
|
}
|
|
switch_event_destroy(&event);
|
|
}
|
|
|
|
if ((flags & ED_DTMF) && switch_channel_has_dtmf(channel)) {
|
|
switch_dtmf_t dtmf = { 0 };
|
|
switch_channel_dequeue_dtmf(channel, &dtmf);
|
|
db[0] = dtmf.digit;
|
|
fcommand = db;
|
|
}
|
|
|
|
if (switch_core_media_bug_test_flag(bug, SMBF_READ_VIDEO_STREAM) ||
|
|
switch_core_media_bug_test_flag(bug, SMBF_WRITE_VIDEO_STREAM)) {
|
|
vid_dual = 1;
|
|
}
|
|
|
|
if (vid_dual || switch_core_media_bug_test_flag(bug, SMBF_READ_VIDEO_PING)) {
|
|
vid_bug = 1;
|
|
}
|
|
|
|
if (fcommand) {
|
|
char *d;
|
|
for (d = fcommand; *d; d++) {
|
|
int z = 1;
|
|
|
|
switch (*d) {
|
|
case '1':
|
|
switch_set_flag(ep, ED_MUX_READ);
|
|
switch_clear_flag(ep, ED_MUX_WRITE);
|
|
if (vid_bug) {
|
|
switch_core_media_bug_set_flag(bug, SMBF_SPY_VIDEO_STREAM);
|
|
switch_core_media_bug_clear_flag(bug, SMBF_SPY_VIDEO_STREAM_BLEG);
|
|
switch_core_session_request_video_refresh(tsession);
|
|
}
|
|
break;
|
|
case '2':
|
|
switch_set_flag(ep, ED_MUX_WRITE);
|
|
switch_clear_flag(ep, ED_MUX_READ);
|
|
if (vid_bug) {
|
|
switch_core_media_bug_set_flag(bug, SMBF_SPY_VIDEO_STREAM_BLEG);
|
|
switch_core_media_bug_clear_flag(bug, SMBF_SPY_VIDEO_STREAM);
|
|
switch_core_session_request_video_refresh(tsession);
|
|
}
|
|
break;
|
|
case '3':
|
|
switch_set_flag(ep, ED_MUX_READ);
|
|
switch_set_flag(ep, ED_MUX_WRITE);
|
|
if (vid_bug) {
|
|
switch_core_media_bug_set_flag(bug, SMBF_SPY_VIDEO_STREAM);
|
|
switch_core_media_bug_set_flag(bug, SMBF_SPY_VIDEO_STREAM_BLEG);
|
|
switch_core_session_request_video_refresh(tsession);
|
|
}
|
|
break;
|
|
|
|
case '4':
|
|
switch_media_bug_set_spy_fmt(bug, switch_media_bug_parse_spy_fmt("dual-crop"));
|
|
break;
|
|
case '5':
|
|
switch_media_bug_set_spy_fmt(bug, switch_media_bug_parse_spy_fmt("lower-right-small"));
|
|
break;
|
|
case '6':
|
|
switch_media_bug_set_spy_fmt(bug, switch_media_bug_parse_spy_fmt("lower-right-large"));
|
|
break;
|
|
case '0':
|
|
switch_clear_flag(ep, ED_MUX_READ);
|
|
switch_clear_flag(ep, ED_MUX_WRITE);
|
|
if (vid_bug) {
|
|
switch_core_media_bug_clear_flag(bug, SMBF_SPY_VIDEO_STREAM);
|
|
switch_core_media_bug_clear_flag(bug, SMBF_SPY_VIDEO_STREAM_BLEG);
|
|
switch_core_session_request_video_refresh(tsession);
|
|
}
|
|
break;
|
|
case '*':
|
|
goto end_loop;
|
|
default:
|
|
z = 0;
|
|
break;
|
|
|
|
}
|
|
|
|
if (z) {
|
|
if (ep->r_buffer) {
|
|
switch_buffer_lock(ep->r_buffer);
|
|
switch_buffer_zero(ep->r_buffer);
|
|
switch_buffer_unlock(ep->r_buffer);
|
|
}
|
|
|
|
if (ep->w_buffer) {
|
|
switch_buffer_lock(ep->w_buffer);
|
|
switch_buffer_zero(ep->w_buffer);
|
|
switch_buffer_unlock(ep->w_buffer);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
if (ep->r_buffer && ep->w_buffer && !switch_test_flag(read_frame, SFF_CNG)) {
|
|
switch_buffer_lock(ep->r_buffer);
|
|
switch_buffer_zwrite(ep->r_buffer, read_frame->data, read_frame->datalen);
|
|
switch_buffer_unlock(ep->r_buffer);
|
|
|
|
switch_buffer_lock(ep->w_buffer);
|
|
switch_buffer_zwrite(ep->w_buffer, read_frame->data, read_frame->datalen);
|
|
switch_buffer_unlock(ep->w_buffer);
|
|
}
|
|
|
|
channels = switch_core_media_bug_test_flag(bug, SMBF_STEREO) ? 2 : tread_impl.number_of_channels;
|
|
|
|
if (channels == 0) {
|
|
channels = 1;
|
|
}
|
|
|
|
tlen = ep->read_impl.decoded_bytes_per_packet * channels;
|
|
|
|
if (len > tlen) {
|
|
len = tlen;
|
|
}
|
|
|
|
if (buffered) {
|
|
buff_min_len = lcm * 2;
|
|
if (switch_buffer_inuse(ep->buffer) < buff_min_len) {
|
|
continue;
|
|
}
|
|
} else {
|
|
buff_min_len = len;
|
|
}
|
|
|
|
if (ep->buffer) {
|
|
switch_buffer_lock(ep->buffer);
|
|
while (switch_buffer_inuse(ep->buffer) >= buff_min_len) {
|
|
int tchanged = 0, changed = 0;
|
|
|
|
write_frame.datalen = (uint32_t) switch_buffer_read(ep->buffer, buf, len);
|
|
write_frame.samples = write_frame.datalen / 2 / channels;
|
|
write_frame.channels = channels;
|
|
|
|
switch_core_session_get_read_impl(tsession, &tread_impl);
|
|
switch_core_session_get_read_impl(session, &read_impl);
|
|
|
|
if (tread_impl.number_of_channels != ep->tread_impl.number_of_channels ||
|
|
tread_impl.actual_samples_per_second != ep->tread_impl.actual_samples_per_second) {
|
|
tchanged = 1;
|
|
}
|
|
|
|
if (read_impl.number_of_channels != ep->read_impl.number_of_channels ||
|
|
read_impl.actual_samples_per_second != ep->read_impl.actual_samples_per_second) {
|
|
changed = 1;
|
|
}
|
|
|
|
if (changed || tchanged) {
|
|
|
|
if (changed) {
|
|
switch_log_printf(SWITCH_CHANNEL_SESSION_LOG(session), SWITCH_LOG_DEBUG,
|
|
"SPYING CHANNEL CODEC CHANGE FROM %dhz@%dc to %dhz@%dc\n",
|
|
ep->read_impl.actual_samples_per_second,
|
|
ep->read_impl.number_of_channels,
|
|
read_impl.actual_samples_per_second,
|
|
read_impl.number_of_channels);
|
|
|
|
tlen = read_impl.decoded_bytes_per_packet * channels;
|
|
|
|
if (len > tlen) {
|
|
len = tlen;
|
|
}
|
|
|
|
}
|
|
|
|
if (tchanged) {
|
|
switch_log_printf(SWITCH_CHANNEL_SESSION_LOG(session), SWITCH_LOG_DEBUG,
|
|
"SPYED CHANNEL CODEC CHANGE FROM %dhz@%dc to %dhz@%dc\n",
|
|
ep->tread_impl.actual_samples_per_second,
|
|
ep->tread_impl.number_of_channels,
|
|
tread_impl.actual_samples_per_second,
|
|
tread_impl.number_of_channels);
|
|
}
|
|
|
|
|
|
switch_core_codec_destroy(&codec);
|
|
|
|
if (tread_impl.decoded_bytes_per_packet < read_impl.decoded_bytes_per_packet) {
|
|
if (switch_core_codec_init(&codec,
|
|
"L16",
|
|
NULL,
|
|
NULL,
|
|
read_impl.actual_samples_per_second,
|
|
read_impl.microseconds_per_packet / 1000,
|
|
read_impl.number_of_channels,
|
|
SWITCH_CODEC_FLAG_ENCODE | SWITCH_CODEC_FLAG_DECODE,
|
|
NULL, switch_core_session_get_pool(session)) != SWITCH_STATUS_SUCCESS) {
|
|
switch_log_printf(SWITCH_CHANNEL_SESSION_LOG(session), SWITCH_LOG_ERROR, "Cannot init codec\n");
|
|
switch_buffer_unlock(ep->buffer);
|
|
goto end;
|
|
}
|
|
buffered = 1;
|
|
lcm = switch_lcm(tread_impl.decoded_bytes_per_packet, read_impl.decoded_bytes_per_packet);
|
|
} else {
|
|
if (switch_core_codec_init(&codec,
|
|
"L16",
|
|
NULL,
|
|
NULL,
|
|
tread_impl.actual_samples_per_second,
|
|
tread_impl.microseconds_per_packet / 1000,
|
|
tread_impl.number_of_channels,
|
|
SWITCH_CODEC_FLAG_ENCODE | SWITCH_CODEC_FLAG_DECODE,
|
|
NULL, switch_core_session_get_pool(session)) != SWITCH_STATUS_SUCCESS) {
|
|
switch_log_printf(SWITCH_CHANNEL_SESSION_LOG(session), SWITCH_LOG_ERROR, "Cannot init codec\n");
|
|
switch_buffer_unlock(ep->buffer);
|
|
goto end;
|
|
}
|
|
if (buffered == 1) {
|
|
buffered = 0;
|
|
}
|
|
}
|
|
|
|
ep->read_impl = read_impl;
|
|
ep->tread_impl = tread_impl;
|
|
}
|
|
|
|
|
|
if (ep->tread_impl.number_of_channels != ep->read_impl.number_of_channels) {
|
|
uint32_t rlen = write_frame.datalen / 2 / ep->tread_impl.number_of_channels;
|
|
|
|
switch_mux_channels((int16_t *) write_frame.data, rlen, ep->tread_impl.number_of_channels, ep->read_impl.number_of_channels);
|
|
write_frame.datalen = rlen * 2 * ep->read_impl.number_of_channels;
|
|
write_frame.samples = write_frame.datalen / 2;
|
|
}
|
|
|
|
if ((status = switch_core_session_write_frame(session, &write_frame, SWITCH_IO_FLAG_NONE, 0)) != SWITCH_STATUS_SUCCESS) {
|
|
break;
|
|
}
|
|
|
|
switch_buffer_unlock(ep->buffer);
|
|
switch_buffer_lock(ep->buffer);
|
|
|
|
if (ep->tread_impl.decoded_bytes_per_packet == ep->read_impl.decoded_bytes_per_packet) {
|
|
/* push just the number of samples worth of a packet. */
|
|
break;
|
|
}
|
|
}
|
|
switch_buffer_unlock(ep->buffer);
|
|
}
|
|
}
|
|
|
|
end_loop:
|
|
|
|
/* Tell the channel we are no longer going to be in a bridge */
|
|
msg.message_id = SWITCH_MESSAGE_INDICATE_UNBRIDGE;
|
|
switch_core_session_receive_message(session, &msg);
|
|
|
|
status = SWITCH_STATUS_SUCCESS;
|
|
end:
|
|
|
|
if (codec_initialized)
|
|
switch_core_codec_destroy(&codec);
|
|
|
|
if (bug) {
|
|
switch_core_media_bug_remove(tsession, &bug);
|
|
}
|
|
|
|
if (ep) {
|
|
if (ep->buffer) {
|
|
switch_buffer_destroy(&ep->buffer);
|
|
}
|
|
|
|
if (ep->r_buffer) {
|
|
switch_buffer_destroy(&ep->r_buffer);
|
|
}
|
|
|
|
if (ep->w_buffer) {
|
|
switch_buffer_destroy(&ep->w_buffer);
|
|
}
|
|
}
|
|
|
|
switch_core_session_rwunlock(tsession);
|
|
|
|
switch_core_session_reset(session, SWITCH_TRUE, SWITCH_TRUE);
|
|
}
|
|
|
|
return status;
|
|
}
|
|
|
|
static switch_status_t record_helper_create(struct record_helper **rh, switch_core_session_t *session)
|
|
{
|
|
switch_status_t status;
|
|
switch_memory_pool_t *pool;
|
|
struct record_helper *newrh;
|
|
|
|
assert(rh);
|
|
assert(session);
|
|
|
|
if ((status = switch_core_new_memory_pool(&pool)) != SWITCH_STATUS_SUCCESS) {
|
|
return status;
|
|
}
|
|
|
|
if (!(newrh = switch_core_alloc(pool, sizeof(*newrh)))) {
|
|
switch_core_destroy_memory_pool(&pool);
|
|
return SWITCH_STATUS_MEMERR;
|
|
}
|
|
|
|
newrh->helper_pool = pool;
|
|
newrh->recording_session = session;
|
|
|
|
*rh = newrh;
|
|
|
|
return SWITCH_STATUS_SUCCESS;
|
|
}
|
|
|
|
static switch_status_t record_helper_destroy(struct record_helper **rh, switch_core_session_t *session)
|
|
{
|
|
switch_memory_pool_t *pool;
|
|
|
|
assert(rh);
|
|
assert(*rh);
|
|
assert(session);
|
|
|
|
if ((*rh)->recording_session != session) {
|
|
switch_log_printf(SWITCH_CHANNEL_SESSION_LOG(session), SWITCH_LOG_ERROR, "Destroying a record helper of another session!\n");
|
|
}
|
|
|
|
if ((*rh)->native) {
|
|
switch_core_file_close(&(*rh)->in_fh);
|
|
switch_core_file_close(&(*rh)->out_fh);
|
|
} else if((*rh)->fh) {
|
|
switch_core_file_close((*rh)->fh);
|
|
}
|
|
|
|
switch_event_safe_destroy((*rh)->variables);
|
|
|
|
pool = (*rh)->helper_pool;
|
|
switch_core_destroy_memory_pool(&pool);
|
|
*rh = NULL;
|
|
|
|
return SWITCH_STATUS_SUCCESS;
|
|
}
|
|
|
|
static const char *get_recording_var(switch_channel_t *channel, switch_event_t *vars, const char *name)
|
|
{
|
|
const char *val = NULL;
|
|
if (!vars || !(val = switch_event_get_header(vars, name))) {
|
|
val = switch_channel_get_variable(channel, name);
|
|
}
|
|
return val;
|
|
}
|
|
|
|
static int recording_var_true(switch_channel_t *channel, switch_event_t *vars, const char *name)
|
|
{
|
|
return switch_true(get_recording_var(channel, vars, name));
|
|
}
|
|
|
|
SWITCH_DECLARE(switch_status_t) switch_ivr_record_session_event(switch_core_session_t *session, const char *file, uint32_t limit, switch_file_handle_t *fh, switch_event_t *vars)
|
|
{
|
|
switch_channel_t *channel = switch_core_session_get_channel(session);
|
|
const char *p;
|
|
const char *vval;
|
|
switch_media_bug_t *bug;
|
|
switch_status_t status;
|
|
time_t to = 0;
|
|
switch_media_bug_flag_t flags = SMBF_READ_STREAM | SMBF_WRITE_STREAM | SMBF_READ_PING;
|
|
uint8_t channels;
|
|
switch_codec_implementation_t read_impl = { 0 };
|
|
struct record_helper *rh = NULL;
|
|
int file_flags = SWITCH_FILE_FLAG_WRITE | SWITCH_FILE_DATA_SHORT;
|
|
switch_bool_t hangup_on_error = SWITCH_FALSE;
|
|
char *file_path = NULL;
|
|
char *ext;
|
|
char *in_file = NULL, *out_file = NULL;
|
|
|
|
if ((p = get_recording_var(channel, vars, "RECORD_HANGUP_ON_ERROR"))) {
|
|
hangup_on_error = switch_true(p);
|
|
}
|
|
|
|
if (switch_channel_pre_answer(channel) != SWITCH_STATUS_SUCCESS) {
|
|
return SWITCH_STATUS_FALSE;
|
|
}
|
|
|
|
if (!switch_channel_media_up(channel) || !switch_core_session_get_read_codec(session)) {
|
|
switch_log_printf(SWITCH_CHANNEL_SESSION_LOG(session), SWITCH_LOG_ERROR, "Can not record session. Media not enabled on channel\n");
|
|
return SWITCH_STATUS_FALSE;
|
|
}
|
|
|
|
switch_core_session_get_read_impl(session, &read_impl);
|
|
channels = read_impl.number_of_channels;
|
|
|
|
if ((bug = switch_channel_get_private(channel, file))) {
|
|
if (recording_var_true(channel, vars, "RECORD_TOGGLE_ON_REPEAT")) {
|
|
return switch_ivr_stop_record_session(session, file);
|
|
}
|
|
|
|
switch_log_printf(SWITCH_CHANNEL_SESSION_LOG(session), SWITCH_LOG_WARNING, "Already recording [%s]\n", file);
|
|
return SWITCH_STATUS_SUCCESS;
|
|
}
|
|
|
|
|
|
if (recording_var_true(channel, vars, "RECORD_CHECK_BRIDGE")) {
|
|
switch_core_session_t *other_session;
|
|
int exist = 0;
|
|
switch_status_t rstatus = SWITCH_STATUS_SUCCESS;
|
|
|
|
if (switch_core_session_get_partner(session, &other_session) == SWITCH_STATUS_SUCCESS) {
|
|
switch_channel_t *other_channel = switch_core_session_get_channel(other_session);
|
|
if ((bug = switch_channel_get_private(other_channel, file))) {
|
|
if (switch_true(switch_channel_get_variable(other_channel, "RECORD_TOGGLE_ON_REPEAT"))) {
|
|
rstatus = switch_ivr_stop_record_session(other_session, file);
|
|
} else {
|
|
switch_log_printf(SWITCH_CHANNEL_SESSION_LOG(other_session), SWITCH_LOG_WARNING, "Already recording [%s]\n", file);
|
|
}
|
|
exist = 1;
|
|
}
|
|
switch_core_session_rwunlock(other_session);
|
|
}
|
|
|
|
if (exist) {
|
|
return rstatus;
|
|
}
|
|
}
|
|
|
|
if ((status = record_helper_create(&rh, session)) != SWITCH_STATUS_SUCCESS) {
|
|
return status;
|
|
}
|
|
|
|
if (fh) {
|
|
switch_file_handle_t *newfh;
|
|
|
|
if ((status = switch_core_file_handle_dup(fh, &newfh, rh->helper_pool)) != SWITCH_STATUS_SUCCESS) {
|
|
switch_goto_status(status, err);
|
|
}
|
|
|
|
fh = newfh;
|
|
} else {
|
|
if (!(fh = switch_core_alloc(rh->helper_pool, sizeof(*fh)))) {
|
|
switch_goto_status(SWITCH_STATUS_MEMERR, err);
|
|
}
|
|
}
|
|
|
|
if (recording_var_true(channel, vars, "RECORD_WRITE_ONLY")) {
|
|
flags &= ~SMBF_READ_STREAM;
|
|
flags |= SMBF_WRITE_STREAM;
|
|
}
|
|
|
|
if (recording_var_true(channel, vars, "RECORD_READ_ONLY")) {
|
|
flags &= ~SMBF_WRITE_STREAM;
|
|
flags |= SMBF_READ_STREAM;
|
|
}
|
|
|
|
if (channels == 1) { /* if leg is already stereo this feature is not available */
|
|
if (recording_var_true(channel, vars, "RECORD_STEREO")) {
|
|
flags |= SMBF_STEREO;
|
|
flags &= ~SMBF_STEREO_SWAP;
|
|
channels = 2;
|
|
}
|
|
|
|
if (recording_var_true(channel, vars, "RECORD_STEREO_SWAP")) {
|
|
flags |= SMBF_STEREO;
|
|
flags |= SMBF_STEREO_SWAP;
|
|
channels = 2;
|
|
}
|
|
}
|
|
|
|
if (recording_var_true(channel, vars, "RECORD_ANSWER_REQ")) {
|
|
flags |= SMBF_ANSWER_REQ;
|
|
}
|
|
|
|
if (recording_var_true(channel, vars, "RECORD_BRIDGE_REQ")) {
|
|
flags |= SMBF_BRIDGE_REQ;
|
|
}
|
|
|
|
if (recording_var_true(channel, vars, "RECORD_APPEND")) {
|
|
file_flags |= SWITCH_FILE_WRITE_APPEND;
|
|
}
|
|
|
|
fh->samplerate = 0;
|
|
if ((vval = get_recording_var(channel, vars, "record_sample_rate"))) {
|
|
int tmp = 0;
|
|
|
|
tmp = atoi(vval);
|
|
|
|
if (switch_is_valid_rate(tmp)) {
|
|
fh->samplerate = tmp;
|
|
}
|
|
}
|
|
|
|
if (!fh->samplerate) {
|
|
fh->samplerate = read_impl.actual_samples_per_second;
|
|
}
|
|
|
|
fh->channels = channels;
|
|
|
|
if ((vval = get_recording_var(channel, vars, "enable_file_write_buffering"))) {
|
|
int tmp = atoi(vval);
|
|
|
|
if (tmp > 0) {
|
|
fh->pre_buffer_datalen = tmp;
|
|
} else if (switch_true(vval)) {
|
|
fh->pre_buffer_datalen = SWITCH_DEFAULT_FILE_BUFFER_LEN;
|
|
}
|
|
|
|
} else {
|
|
fh->pre_buffer_datalen = SWITCH_DEFAULT_FILE_BUFFER_LEN;
|
|
}
|
|
|
|
if (!switch_is_file_path(file)) {
|
|
char *tfile = NULL;
|
|
char *e;
|
|
const char *prefix;
|
|
|
|
prefix = get_recording_var(channel, vars, "sound_prefix");
|
|
|
|
if (!prefix) {
|
|
prefix = SWITCH_GLOBAL_dirs.base_dir;
|
|
}
|
|
|
|
if (*file == '[') {
|
|
tfile = switch_core_strdup(rh->helper_pool, file);
|
|
if ((e = switch_find_end_paren(tfile, '[', ']'))) {
|
|
*e = '\0';
|
|
file = e + 1;
|
|
} else {
|
|
tfile = NULL;
|
|
}
|
|
} else {
|
|
file_path = switch_core_sprintf(rh->helper_pool, "%s%s%s", prefix, SWITCH_PATH_SEPARATOR, file);
|
|
}
|
|
|
|
file = switch_core_sprintf(rh->helper_pool, "%s%s%s%s%s", switch_str_nil(tfile), tfile ? "]" : "", prefix, SWITCH_PATH_SEPARATOR, file);
|
|
} else {
|
|
file_path = switch_core_strdup(rh->helper_pool, file);
|
|
}
|
|
|
|
if (file_path && !strstr(file_path, SWITCH_URL_SEPARATOR)) {
|
|
char *p;
|
|
char *path = switch_core_strdup(rh->helper_pool, file_path);
|
|
|
|
if ((p = strrchr(path, *SWITCH_PATH_SEPARATOR))) {
|
|
*p = '\0';
|
|
|
|
if (*path == '{') {
|
|
path = switch_find_end_paren(path, '{', '}') + 1;
|
|
}
|
|
|
|
if (switch_dir_make_recursive(path, SWITCH_DEFAULT_DIR_PERMS, switch_core_session_get_pool(session)) != SWITCH_STATUS_SUCCESS) {
|
|
switch_log_printf(SWITCH_CHANNEL_SESSION_LOG(session), SWITCH_LOG_ERROR, "Error creating %s\n", path);
|
|
set_completion_cause(rh, "uri-failure");
|
|
switch_goto_status(SWITCH_STATUS_GENERR, err);
|
|
}
|
|
|
|
} else {
|
|
switch_log_printf(SWITCH_CHANNEL_SESSION_LOG(session), SWITCH_LOG_ERROR, "Error finding the folder path section in '%s'\n", path);
|
|
path = NULL;
|
|
}
|
|
}
|
|
|
|
if ((ext = strrchr(file, '.'))) {
|
|
ext++;
|
|
|
|
if (switch_channel_test_flag(channel, CF_VIDEO)) {
|
|
file_flags |= SWITCH_FILE_FLAG_VIDEO;
|
|
}
|
|
|
|
if (switch_core_file_open(fh, file, channels, read_impl.actual_samples_per_second, file_flags, NULL) != SWITCH_STATUS_SUCCESS) {
|
|
switch_log_printf(SWITCH_CHANNEL_SESSION_LOG(session), SWITCH_LOG_ERROR, "Error opening %s\n", file);
|
|
if (hangup_on_error) {
|
|
switch_channel_hangup(channel, SWITCH_CAUSE_DESTINATION_OUT_OF_ORDER);
|
|
switch_core_session_reset(session, SWITCH_TRUE, SWITCH_TRUE);
|
|
}
|
|
set_completion_cause(rh, "uri-failure");
|
|
switch_goto_status(SWITCH_STATUS_GENERR, err);
|
|
}
|
|
|
|
if (fh->params) {
|
|
if ((p = switch_event_get_header(fh->params,"record_write_only")) && switch_true(p)) {
|
|
flags &= ~SMBF_READ_STREAM;
|
|
flags |= SMBF_WRITE_STREAM;
|
|
}
|
|
|
|
|
|
if ((p = switch_event_get_header(fh->params, "record_read_only")) && switch_true(p)) {
|
|
flags &= ~SMBF_WRITE_STREAM;
|
|
flags |= SMBF_READ_STREAM;
|
|
}
|
|
}
|
|
|
|
if (switch_core_file_has_video(fh, SWITCH_TRUE)) {
|
|
//switch_core_media_set_video_file(session, fh, SWITCH_RW_READ);
|
|
//switch_channel_set_flag_recursive(session->channel, CF_VIDEO_DECODED_READ);
|
|
|
|
if (recording_var_true(channel, vars, "record_concat_video")) {
|
|
flags |= SMBF_READ_VIDEO_STREAM;
|
|
flags |= SMBF_WRITE_VIDEO_STREAM;
|
|
} else if (recording_var_true(channel, vars, "record_bleg_video")) {
|
|
flags |= SMBF_WRITE_VIDEO_STREAM;
|
|
} else {
|
|
flags |= SMBF_READ_VIDEO_STREAM;
|
|
}
|
|
} else {
|
|
flags &= ~SMBF_READ_VIDEO_PING;
|
|
flags &= ~SMBF_READ_VIDEO_STREAM;
|
|
flags &= ~SMBF_WRITE_VIDEO_STREAM;
|
|
}
|
|
|
|
} else {
|
|
int tflags = 0;
|
|
|
|
ext = read_impl.iananame;
|
|
|
|
in_file = switch_core_sprintf(rh->helper_pool, "%s-in.%s", file, ext);
|
|
out_file = switch_core_sprintf(rh->helper_pool, "%s-out.%s", file, ext);
|
|
rh->in_fh.pre_buffer_datalen = rh->out_fh.pre_buffer_datalen = fh->pre_buffer_datalen;
|
|
channels = 1;
|
|
switch_set_flag(&rh->in_fh, SWITCH_FILE_NATIVE);
|
|
switch_set_flag(&rh->out_fh, SWITCH_FILE_NATIVE);
|
|
|
|
if (switch_core_file_open(&rh->in_fh, in_file, channels, read_impl.actual_samples_per_second, file_flags, NULL) != SWITCH_STATUS_SUCCESS) {
|
|
switch_log_printf(SWITCH_CHANNEL_SESSION_LOG(session), SWITCH_LOG_ERROR, "Error opening %s\n", in_file);
|
|
if (hangup_on_error) {
|
|
switch_channel_hangup(channel, SWITCH_CAUSE_DESTINATION_OUT_OF_ORDER);
|
|
switch_core_session_reset(session, SWITCH_TRUE, SWITCH_TRUE);
|
|
}
|
|
set_completion_cause(rh, "uri-failure");
|
|
switch_goto_status(SWITCH_STATUS_GENERR, err);
|
|
}
|
|
|
|
if (switch_core_file_open(&rh->out_fh, out_file, channels, read_impl.actual_samples_per_second, file_flags, NULL) != SWITCH_STATUS_SUCCESS) {
|
|
switch_log_printf(SWITCH_CHANNEL_SESSION_LOG(session), SWITCH_LOG_ERROR, "Error opening %s\n", out_file);
|
|
switch_core_file_close(&rh->in_fh);
|
|
if (hangup_on_error) {
|
|
switch_channel_hangup(channel, SWITCH_CAUSE_DESTINATION_OUT_OF_ORDER);
|
|
switch_core_session_reset(session, SWITCH_TRUE, SWITCH_TRUE);
|
|
}
|
|
set_completion_cause(rh, "uri-failure");
|
|
switch_goto_status(SWITCH_STATUS_GENERR, err);
|
|
}
|
|
|
|
rh->native = 1;
|
|
fh = NULL;
|
|
|
|
if ((flags & SMBF_WRITE_STREAM)) {
|
|
tflags |= SMBF_TAP_NATIVE_WRITE;
|
|
}
|
|
|
|
if ((flags & SMBF_READ_STREAM)) {
|
|
tflags |= SMBF_TAP_NATIVE_READ;
|
|
}
|
|
|
|
flags = tflags;
|
|
}
|
|
|
|
if ((p = get_recording_var(channel, vars, "RECORD_TITLE"))) {
|
|
vval = (const char *) switch_core_strdup(rh->helper_pool, p);
|
|
if (fh) switch_core_file_set_string(fh, SWITCH_AUDIO_COL_STR_TITLE, vval);
|
|
switch_channel_set_variable(channel, "RECORD_TITLE", NULL);
|
|
}
|
|
|
|
if ((p = get_recording_var(channel, vars, "RECORD_COPYRIGHT"))) {
|
|
vval = (const char *) switch_core_strdup(rh->helper_pool, p);
|
|
if (fh) switch_core_file_set_string(fh, SWITCH_AUDIO_COL_STR_COPYRIGHT, vval);
|
|
switch_channel_set_variable(channel, "RECORD_COPYRIGHT", NULL);
|
|
}
|
|
|
|
if ((p = get_recording_var(channel, vars, "RECORD_SOFTWARE"))) {
|
|
vval = (const char *) switch_core_strdup(rh->helper_pool, p);
|
|
if (fh) switch_core_file_set_string(fh, SWITCH_AUDIO_COL_STR_SOFTWARE, vval);
|
|
switch_channel_set_variable(channel, "RECORD_SOFTWARE", NULL);
|
|
}
|
|
|
|
if ((p = get_recording_var(channel, vars, "RECORD_ARTIST"))) {
|
|
vval = (const char *) switch_core_strdup(rh->helper_pool, p);
|
|
if (fh) switch_core_file_set_string(fh, SWITCH_AUDIO_COL_STR_ARTIST, vval);
|
|
switch_channel_set_variable(channel, "RECORD_ARTIST", NULL);
|
|
}
|
|
|
|
if ((p = get_recording_var(channel, vars, "RECORD_COMMENT"))) {
|
|
vval = (const char *) switch_core_strdup(rh->helper_pool, p);
|
|
if (fh) switch_core_file_set_string(fh, SWITCH_AUDIO_COL_STR_COMMENT, vval);
|
|
switch_channel_set_variable(channel, "RECORD_COMMENT", NULL);
|
|
}
|
|
|
|
if ((p = get_recording_var(channel, vars, "RECORD_DATE"))) {
|
|
vval = (const char *) switch_core_strdup(rh->helper_pool, p);
|
|
if (fh) switch_core_file_set_string(fh, SWITCH_AUDIO_COL_STR_DATE, vval);
|
|
switch_channel_set_variable(channel, "RECORD_DATE", NULL);
|
|
}
|
|
|
|
if (limit) {
|
|
to = switch_epoch_time_now(NULL) + limit;
|
|
}
|
|
|
|
rh->fh = fh;
|
|
rh->file = switch_core_strdup(rh->helper_pool, file);
|
|
rh->packet_len = read_impl.decoded_bytes_per_packet;
|
|
|
|
if (file_flags & SWITCH_FILE_WRITE_APPEND) {
|
|
rh->min_sec = 3;
|
|
}
|
|
|
|
if ((p = get_recording_var(channel, vars, "RECORD_MIN_SEC"))) {
|
|
int tmp = atoi(p);
|
|
if (tmp >= 0) {
|
|
rh->min_sec = tmp;
|
|
}
|
|
}
|
|
|
|
if ((p = get_recording_var(channel, vars, "RECORD_INITIAL_TIMEOUT_MS"))) {
|
|
int tmp = atoi(p);
|
|
if (tmp >= 0) {
|
|
rh->initial_timeout_ms = tmp;
|
|
rh->silence_threshold = 200;
|
|
}
|
|
}
|
|
|
|
if ((p = get_recording_var(channel, vars, "RECORD_FINAL_TIMEOUT_MS"))) {
|
|
int tmp = atoi(p);
|
|
if (tmp >= 0) {
|
|
rh->final_timeout_ms = tmp;
|
|
rh->silence_threshold = 200;
|
|
}
|
|
}
|
|
|
|
if ((p = get_recording_var(channel, vars, "RECORD_SILENCE_THRESHOLD"))) {
|
|
int tmp = atoi(p);
|
|
if (tmp >= 0) {
|
|
rh->silence_threshold = tmp;
|
|
}
|
|
}
|
|
|
|
if (vars) {
|
|
switch_event_dup(&rh->variables, vars);
|
|
}
|
|
|
|
rh->hangup_on_error = hangup_on_error;
|
|
|
|
if ((status = switch_core_media_bug_add(session, "session_record", file,
|
|
record_callback, rh, to, flags, &bug)) != SWITCH_STATUS_SUCCESS) {
|
|
if (switch_channel_ready(channel)) {
|
|
switch_log_printf(SWITCH_CHANNEL_SESSION_LOG(session), SWITCH_LOG_ERROR, "Error adding media bug for file %s\n", file);
|
|
} else {
|
|
switch_log_printf(SWITCH_CHANNEL_SESSION_LOG(session), SWITCH_LOG_WARNING, "Error adding media bug for file %s\n", file);
|
|
}
|
|
switch_goto_status(status, err);
|
|
}
|
|
|
|
if ((p = get_recording_var(channel, vars, "RECORD_PRE_BUFFER_FRAMES"))) {
|
|
int tmp = atoi(p);
|
|
|
|
if (tmp > 0) {
|
|
switch_core_media_bug_set_pre_buffer_framecount(bug, tmp);
|
|
}
|
|
} else {
|
|
switch_core_media_bug_set_pre_buffer_framecount(bug, 25);
|
|
}
|
|
|
|
switch_channel_set_private(channel, file, bug);
|
|
|
|
if (switch_channel_test_flag(channel, CF_VIDEO)) {
|
|
switch_core_session_message_t msg = { 0 };
|
|
|
|
msg.from = __FILE__;
|
|
msg.message_id = SWITCH_MESSAGE_INDICATE_VIDEO_REFRESH_REQ;
|
|
|
|
switch_core_session_receive_message(session, &msg);
|
|
}
|
|
|
|
return SWITCH_STATUS_SUCCESS;
|
|
|
|
err:
|
|
if (!zstr(rh->completion_cause)) {
|
|
switch_channel_set_variable_printf(channel, "record_completion_cause", "%s", rh->completion_cause);
|
|
}
|
|
record_helper_destroy(&rh, session);
|
|
|
|
return status;
|
|
}
|
|
|
|
SWITCH_DECLARE(switch_status_t) switch_ivr_record_session(switch_core_session_t *session, const char *file, uint32_t limit, switch_file_handle_t *fh)
|
|
{
|
|
return switch_ivr_record_session_event(session, file, limit, fh, NULL);
|
|
}
|
|
|
|
typedef struct {
|
|
SpeexPreprocessState *read_st;
|
|
SpeexPreprocessState *write_st;
|
|
SpeexEchoState *read_ec;
|
|
SpeexEchoState *write_ec;
|
|
switch_byte_t read_data[2048];
|
|
switch_byte_t write_data[2048];
|
|
switch_byte_t read_out[2048];
|
|
switch_byte_t write_out[2048];
|
|
switch_mutex_t *read_mutex;
|
|
switch_mutex_t *write_mutex;
|
|
int done;
|
|
} pp_cb_t;
|
|
|
|
static switch_bool_t preprocess_callback(switch_media_bug_t *bug, void *user_data, switch_abc_type_t type)
|
|
{
|
|
switch_core_session_t *session = switch_core_media_bug_get_session(bug);
|
|
switch_channel_t *channel = switch_core_session_get_channel(session);
|
|
pp_cb_t *cb = (pp_cb_t *) user_data;
|
|
switch_codec_implementation_t read_impl = { 0 };
|
|
switch_frame_t *frame = NULL;
|
|
|
|
switch_core_session_get_read_impl(session, &read_impl);
|
|
|
|
switch (type) {
|
|
case SWITCH_ABC_TYPE_INIT:
|
|
{
|
|
switch_mutex_init(&cb->read_mutex, SWITCH_MUTEX_NESTED, switch_core_session_get_pool(session));
|
|
switch_mutex_init(&cb->write_mutex, SWITCH_MUTEX_NESTED, switch_core_session_get_pool(session));
|
|
}
|
|
break;
|
|
case SWITCH_ABC_TYPE_CLOSE:
|
|
{
|
|
if (cb->read_st) {
|
|
speex_preprocess_state_destroy(cb->read_st);
|
|
}
|
|
|
|
if (cb->write_st) {
|
|
speex_preprocess_state_destroy(cb->write_st);
|
|
}
|
|
|
|
if (cb->read_ec) {
|
|
speex_echo_state_destroy(cb->read_ec);
|
|
}
|
|
|
|
if (cb->write_ec) {
|
|
speex_echo_state_destroy(cb->write_ec);
|
|
}
|
|
|
|
switch_channel_set_private(channel, "_preprocess", NULL);
|
|
}
|
|
break;
|
|
case SWITCH_ABC_TYPE_READ_REPLACE:
|
|
{
|
|
if (cb->done)
|
|
return SWITCH_FALSE;
|
|
frame = switch_core_media_bug_get_read_replace_frame(bug);
|
|
|
|
if (cb->read_st) {
|
|
|
|
if (cb->read_ec) {
|
|
speex_echo_cancellation(cb->read_ec, (int16_t *) frame->data, (int16_t *) cb->write_data, (int16_t *) cb->read_out);
|
|
memcpy(frame->data, cb->read_out, frame->datalen);
|
|
}
|
|
|
|
speex_preprocess_run(cb->read_st, frame->data);
|
|
}
|
|
|
|
if (cb->write_ec) {
|
|
memcpy(cb->read_data, frame->data, frame->datalen);
|
|
}
|
|
}
|
|
break;
|
|
case SWITCH_ABC_TYPE_WRITE_REPLACE:
|
|
{
|
|
if (cb->done)
|
|
return SWITCH_FALSE;
|
|
frame = switch_core_media_bug_get_write_replace_frame(bug);
|
|
|
|
if (cb->write_st) {
|
|
|
|
if (cb->write_ec) {
|
|
speex_echo_cancellation(cb->write_ec, (int16_t *) frame->data, (int16_t *) cb->read_data, (int16_t *) cb->write_out);
|
|
memcpy(frame->data, cb->write_out, frame->datalen);
|
|
}
|
|
|
|
speex_preprocess_run(cb->write_st, frame->data);
|
|
}
|
|
|
|
if (cb->read_ec) {
|
|
memcpy(cb->write_data, frame->data, frame->datalen);
|
|
}
|
|
}
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
|
|
return SWITCH_TRUE;
|
|
}
|
|
|
|
SWITCH_DECLARE(switch_status_t) switch_ivr_preprocess_session(switch_core_session_t *session, const char *cmds)
|
|
{
|
|
switch_channel_t *channel = switch_core_session_get_channel(session);
|
|
switch_media_bug_t *bug;
|
|
switch_status_t status;
|
|
time_t to = 0;
|
|
switch_media_bug_flag_t flags = SMBF_NO_PAUSE;
|
|
switch_codec_implementation_t read_impl = { 0 };
|
|
pp_cb_t *cb;
|
|
int update = 0;
|
|
int argc;
|
|
char *mydata = NULL, *argv[5];
|
|
int i = 0;
|
|
|
|
switch_core_session_get_read_impl(session, &read_impl);
|
|
|
|
if ((cb = switch_channel_get_private(channel, "_preprocess"))) {
|
|
update = 1;
|
|
} else {
|
|
cb = switch_core_session_alloc(session, sizeof(*cb));
|
|
}
|
|
|
|
|
|
if (update) {
|
|
if (!strcasecmp(cmds, "stop")) {
|
|
cb->done = 1;
|
|
return SWITCH_STATUS_SUCCESS;
|
|
}
|
|
}
|
|
|
|
mydata = strdup(cmds);
|
|
argc = switch_separate_string(mydata, ',', argv, (sizeof(argv) / sizeof(argv[0])));
|
|
|
|
for (i = 0; i < argc; i++) {
|
|
char *var = argv[i];
|
|
char *val = NULL;
|
|
char rw;
|
|
int tr;
|
|
int err = 1;
|
|
SpeexPreprocessState *st = NULL;
|
|
SpeexEchoState *ec = NULL;
|
|
switch_mutex_t *mutex = NULL;
|
|
int r = 0;
|
|
|
|
if (var) {
|
|
if ((val = strchr(var, '='))) {
|
|
*val++ = '\0';
|
|
|
|
rw = *var++;
|
|
while (*var == '.' || *var == '_') {
|
|
var++;
|
|
}
|
|
|
|
if (rw == 'r') {
|
|
if (!cb->read_st) {
|
|
cb->read_st = speex_preprocess_state_init(read_impl.samples_per_packet, read_impl.samples_per_second);
|
|
flags |= SMBF_READ_REPLACE;
|
|
}
|
|
st = cb->read_st;
|
|
ec = cb->read_ec;
|
|
mutex = cb->read_mutex;
|
|
} else if (rw == 'w') {
|
|
if (!cb->write_st) {
|
|
cb->write_st = speex_preprocess_state_init(read_impl.samples_per_packet, read_impl.samples_per_second);
|
|
flags |= SMBF_WRITE_REPLACE;
|
|
}
|
|
st = cb->write_st;
|
|
ec = cb->write_ec;
|
|
mutex = cb->write_mutex;
|
|
}
|
|
|
|
if (mutex)
|
|
switch_mutex_lock(mutex);
|
|
|
|
if (st) {
|
|
err = 0;
|
|
tr = switch_true(val);
|
|
if (!strcasecmp(var, "agc")) {
|
|
int l = read_impl.samples_per_second;
|
|
int tmp = atoi(val);
|
|
|
|
if (!tr) {
|
|
l = tmp;
|
|
}
|
|
|
|
switch_log_printf(SWITCH_CHANNEL_SESSION_LOG(session), SWITCH_LOG_DEBUG, "Setting AGC on %c to %d\n", rw, tr);
|
|
speex_preprocess_ctl(st, SPEEX_PREPROCESS_SET_AGC, &tr);
|
|
speex_preprocess_ctl(st, SPEEX_PREPROCESS_SET_AGC_LEVEL, &l);
|
|
|
|
} else if (!strcasecmp(var, "noise_suppress")) {
|
|
int db = atoi(val);
|
|
if (db < 0) {
|
|
r = speex_preprocess_ctl(st, SPEEX_PREPROCESS_SET_NOISE_SUPPRESS, &db);
|
|
switch_log_printf(SWITCH_CHANNEL_SESSION_LOG(session), SWITCH_LOG_DEBUG, "Setting NOISE_SUPPRESS on %c to %d [%d]\n", rw, db,
|
|
r);
|
|
} else {
|
|
switch_log_printf(SWITCH_CHANNEL_SESSION_LOG(session), SWITCH_LOG_ERROR, "Syntax error noise_suppress should be in -db\n");
|
|
}
|
|
} else if (!strcasecmp(var, "echo_cancel")) {
|
|
int tail = 1024;
|
|
int tmp = atoi(val);
|
|
|
|
if (!tr && tmp > 0) {
|
|
tail = tmp;
|
|
} else if (!tr) {
|
|
if (ec) {
|
|
if (rw == 'r') {
|
|
speex_echo_state_destroy(cb->read_ec);
|
|
cb->read_ec = NULL;
|
|
} else {
|
|
speex_echo_state_destroy(cb->write_ec);
|
|
cb->write_ec = NULL;
|
|
}
|
|
}
|
|
|
|
ec = NULL;
|
|
}
|
|
|
|
if (!ec) {
|
|
if (rw == 'r') {
|
|
ec = cb->read_ec = speex_echo_state_init(read_impl.samples_per_packet, tail);
|
|
speex_echo_ctl(ec, SPEEX_ECHO_SET_SAMPLING_RATE, &read_impl.samples_per_second);
|
|
flags |= SMBF_WRITE_REPLACE;
|
|
} else {
|
|
ec = cb->write_ec = speex_echo_state_init(read_impl.samples_per_packet, tail);
|
|
speex_echo_ctl(ec, SPEEX_ECHO_SET_SAMPLING_RATE, &read_impl.samples_per_second);
|
|
flags |= SMBF_READ_REPLACE;
|
|
}
|
|
speex_preprocess_ctl(st, SPEEX_PREPROCESS_SET_ECHO_STATE, ec);
|
|
}
|
|
|
|
|
|
} else if (!strcasecmp(var, "echo_suppress")) {
|
|
int db = atoi(val);
|
|
if (db < 0) {
|
|
speex_preprocess_ctl(st, SPEEX_PREPROCESS_SET_ECHO_SUPPRESS, &db);
|
|
switch_log_printf(SWITCH_CHANNEL_SESSION_LOG(session), SWITCH_LOG_DEBUG, "Setting ECHO_SUPPRESS on %c to %d [%d]\n", rw, db,
|
|
r);
|
|
} else {
|
|
switch_log_printf(SWITCH_CHANNEL_SESSION_LOG(session), SWITCH_LOG_ERROR, "Syntax error echo_suppress should be in -db\n");
|
|
}
|
|
} else {
|
|
switch_log_printf(SWITCH_CHANNEL_SESSION_LOG(session), SWITCH_LOG_WARNING, "Warning unknown parameter [%s] \n", var);
|
|
}
|
|
}
|
|
}
|
|
|
|
if (mutex)
|
|
switch_mutex_unlock(mutex);
|
|
|
|
if (err) {
|
|
switch_log_printf(SWITCH_CHANNEL_SESSION_LOG(session), SWITCH_LOG_ERROR, "Syntax error parsing preprocessor commands\n");
|
|
}
|
|
|
|
} else {
|
|
break;
|
|
}
|
|
}
|
|
|
|
|
|
switch_safe_free(mydata);
|
|
|
|
if (update) {
|
|
return SWITCH_STATUS_SUCCESS;
|
|
}
|
|
|
|
|
|
if ((status = switch_core_media_bug_add(session, "preprocess", NULL,
|
|
preprocess_callback, cb, to, flags, &bug)) != SWITCH_STATUS_SUCCESS) {
|
|
switch_log_printf(SWITCH_CHANNEL_SESSION_LOG(session), SWITCH_LOG_ERROR, "Error adding media bug.\n");
|
|
if (cb->read_st) {
|
|
speex_preprocess_state_destroy(cb->read_st);
|
|
}
|
|
|
|
if (cb->write_st) {
|
|
speex_preprocess_state_destroy(cb->write_st);
|
|
}
|
|
|
|
if (cb->read_ec) {
|
|
speex_echo_state_destroy(cb->read_ec);
|
|
}
|
|
|
|
if (cb->write_ec) {
|
|
speex_echo_state_destroy(cb->write_ec);
|
|
}
|
|
|
|
return status;
|
|
}
|
|
|
|
switch_channel_set_private(channel, "_preprocess", cb);
|
|
|
|
return SWITCH_STATUS_SUCCESS;
|
|
}
|
|
|
|
|
|
typedef struct {
|
|
switch_core_session_t *session;
|
|
int mute;
|
|
int read_level;
|
|
int write_level;
|
|
int read_mute;
|
|
int write_mute;
|
|
} switch_session_audio_t;
|
|
|
|
static switch_bool_t session_audio_callback(switch_media_bug_t *bug, void *user_data, switch_abc_type_t type)
|
|
{
|
|
switch_session_audio_t *pvt = (switch_session_audio_t *) user_data;
|
|
switch_frame_t *frame = NULL;
|
|
int level = 0, mute = 0;
|
|
switch_core_session_t *session = switch_core_media_bug_get_session(bug);
|
|
switch_codec_implementation_t read_impl = { 0 };
|
|
|
|
switch_core_session_get_read_impl(session, &read_impl);
|
|
|
|
|
|
if (type == SWITCH_ABC_TYPE_READ_REPLACE || type == SWITCH_ABC_TYPE_WRITE_REPLACE) {
|
|
if (!(pvt->read_level || pvt->write_level || pvt->read_mute || pvt->write_mute)) {
|
|
switch_channel_set_private(switch_core_session_get_channel(pvt->session), "__audio", NULL);
|
|
return SWITCH_FALSE;
|
|
}
|
|
}
|
|
|
|
if (type == SWITCH_ABC_TYPE_READ_REPLACE) {
|
|
level = pvt->read_level;
|
|
mute = pvt->read_mute;
|
|
frame = switch_core_media_bug_get_read_replace_frame(bug);
|
|
} else if (type == SWITCH_ABC_TYPE_WRITE_REPLACE) {
|
|
level = pvt->write_level;
|
|
mute = pvt->write_mute;
|
|
frame = switch_core_media_bug_get_write_replace_frame(bug);
|
|
}
|
|
|
|
if (frame) {
|
|
if (mute) {
|
|
if (mute > 1) {
|
|
switch_generate_sln_silence(frame->data, frame->datalen / 2, read_impl.number_of_channels, mute);
|
|
} else {
|
|
memset(frame->data, 0, frame->datalen);
|
|
}
|
|
} else if (level) {
|
|
switch_change_sln_volume(frame->data, frame->datalen / 2, level);
|
|
}
|
|
|
|
if (type == SWITCH_ABC_TYPE_READ_REPLACE) {
|
|
switch_core_media_bug_set_read_replace_frame(bug, frame);
|
|
} else if (type == SWITCH_ABC_TYPE_WRITE_REPLACE) {
|
|
switch_core_media_bug_set_write_replace_frame(bug, frame);
|
|
}
|
|
}
|
|
|
|
return SWITCH_TRUE;
|
|
}
|
|
|
|
SWITCH_DECLARE(switch_status_t) switch_ivr_stop_session_audio(switch_core_session_t *session)
|
|
{
|
|
switch_media_bug_t *bug;
|
|
switch_channel_t *channel = switch_core_session_get_channel(session);
|
|
|
|
if ((bug = switch_channel_get_private(channel, "__audio"))) {
|
|
switch_channel_set_private(channel, "__audio", NULL);
|
|
switch_core_media_bug_remove(session, &bug);
|
|
return SWITCH_STATUS_SUCCESS;
|
|
}
|
|
return SWITCH_STATUS_FALSE;
|
|
}
|
|
|
|
SWITCH_DECLARE(switch_status_t) switch_ivr_session_audio(switch_core_session_t *session, const char *cmd, const char *direction, int level)
|
|
{
|
|
switch_channel_t *channel = switch_core_session_get_channel(session);
|
|
switch_media_bug_t *bug;
|
|
switch_status_t status;
|
|
switch_session_audio_t *pvt;
|
|
switch_codec_implementation_t read_impl = { 0 };
|
|
int existing = 0, c_read = 0, c_write = 0, flags = SMBF_NO_PAUSE;
|
|
|
|
if (switch_channel_pre_answer(channel) != SWITCH_STATUS_SUCCESS) {
|
|
return SWITCH_STATUS_FALSE;
|
|
}
|
|
|
|
switch_core_session_get_read_impl(session, &read_impl);
|
|
|
|
|
|
if ((bug = switch_channel_get_private(channel, "__audio"))) {
|
|
pvt = switch_core_media_bug_get_user_data(bug);
|
|
existing = 1;
|
|
} else {
|
|
if (!(pvt = switch_core_session_alloc(session, sizeof(*pvt)))) {
|
|
return SWITCH_STATUS_MEMERR;
|
|
}
|
|
|
|
pvt->session = session;
|
|
}
|
|
|
|
|
|
if (!strcasecmp(direction, "write")) {
|
|
flags = SMBF_WRITE_REPLACE;
|
|
c_write = 1;
|
|
} else if (!strcasecmp(direction, "read")) {
|
|
flags = SMBF_READ_REPLACE;
|
|
c_read = 1;
|
|
} else if (!strcasecmp(direction, "both")) {
|
|
flags = SMBF_READ_REPLACE | SMBF_WRITE_REPLACE;
|
|
c_read = c_write = 1;
|
|
}
|
|
|
|
|
|
if (!strcasecmp(cmd, "mute")) {
|
|
if (c_read) {
|
|
pvt->read_mute = level;
|
|
pvt->read_level = 0;
|
|
}
|
|
if (c_write) {
|
|
pvt->write_mute = level;
|
|
pvt->write_level = 0;
|
|
}
|
|
} else if (!strcasecmp(cmd, "level")) {
|
|
if (level < 5 && level > -5) {
|
|
if (c_read) {
|
|
pvt->read_level = level;
|
|
}
|
|
if (c_write) {
|
|
pvt->write_level = level;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (existing) {
|
|
switch_core_media_bug_set_flag(bug, flags);
|
|
} else {
|
|
if ((status = switch_core_media_bug_add(session, "audio", cmd,
|
|
session_audio_callback, pvt, 0, flags, &bug)) != SWITCH_STATUS_SUCCESS) {
|
|
return status;
|
|
}
|
|
|
|
switch_channel_set_private(channel, "__audio", bug);
|
|
}
|
|
|
|
|
|
return SWITCH_STATUS_SUCCESS;
|
|
}
|
|
|
|
|
|
typedef struct {
|
|
switch_core_session_t *session;
|
|
teletone_dtmf_detect_state_t dtmf_detect;
|
|
} switch_inband_dtmf_t;
|
|
|
|
static switch_bool_t inband_dtmf_callback(switch_media_bug_t *bug, void *user_data, switch_abc_type_t type)
|
|
{
|
|
switch_inband_dtmf_t *pvt = (switch_inband_dtmf_t *) user_data;
|
|
switch_frame_t *frame = NULL;
|
|
switch_channel_t *channel = switch_core_session_get_channel(pvt->session);
|
|
|
|
switch (type) {
|
|
case SWITCH_ABC_TYPE_INIT:
|
|
break;
|
|
case SWITCH_ABC_TYPE_CLOSE:
|
|
break;
|
|
case SWITCH_ABC_TYPE_READ_REPLACE:
|
|
if ((frame = switch_core_media_bug_get_read_replace_frame(bug))) {
|
|
if (teletone_dtmf_detect(&pvt->dtmf_detect, frame->data, frame->samples) == TT_HIT_END) {
|
|
switch_dtmf_t dtmf = {0};
|
|
|
|
teletone_dtmf_get(&pvt->dtmf_detect, &dtmf.digit, &dtmf.duration);
|
|
switch_log_printf(SWITCH_CHANNEL_SESSION_LOG(switch_core_media_bug_get_session(bug)), SWITCH_LOG_DEBUG, "DTMF DETECTED: [%c][%d]\n",
|
|
dtmf.digit, dtmf.duration);
|
|
dtmf.source = SWITCH_DTMF_INBAND_AUDIO;
|
|
switch_channel_queue_dtmf(channel, &dtmf);
|
|
}
|
|
switch_core_media_bug_set_read_replace_frame(bug, frame);
|
|
}
|
|
break;
|
|
case SWITCH_ABC_TYPE_WRITE:
|
|
default:
|
|
break;
|
|
}
|
|
|
|
return SWITCH_TRUE;
|
|
}
|
|
|
|
SWITCH_DECLARE(switch_status_t) switch_ivr_stop_inband_dtmf_session(switch_core_session_t *session)
|
|
{
|
|
switch_media_bug_t *bug;
|
|
switch_channel_t *channel = switch_core_session_get_channel(session);
|
|
|
|
if ((bug = switch_channel_get_private(channel, "dtmf"))) {
|
|
switch_channel_set_private(channel, "dtmf", NULL);
|
|
switch_core_media_bug_remove(session, &bug);
|
|
return SWITCH_STATUS_SUCCESS;
|
|
}
|
|
return SWITCH_STATUS_FALSE;
|
|
}
|
|
|
|
SWITCH_DECLARE(switch_status_t) switch_ivr_inband_dtmf_session(switch_core_session_t *session)
|
|
{
|
|
switch_channel_t *channel = switch_core_session_get_channel(session);
|
|
switch_media_bug_t *bug;
|
|
switch_status_t status;
|
|
switch_inband_dtmf_t *pvt;
|
|
switch_codec_implementation_t read_impl = { 0 };
|
|
|
|
switch_core_session_get_read_impl(session, &read_impl);
|
|
|
|
if (!(pvt = switch_core_session_alloc(session, sizeof(*pvt)))) {
|
|
return SWITCH_STATUS_MEMERR;
|
|
}
|
|
|
|
teletone_dtmf_detect_init(&pvt->dtmf_detect, read_impl.actual_samples_per_second);
|
|
|
|
pvt->session = session;
|
|
|
|
|
|
if (switch_channel_pre_answer(channel) != SWITCH_STATUS_SUCCESS) {
|
|
return SWITCH_STATUS_FALSE;
|
|
}
|
|
|
|
if ((status = switch_core_media_bug_add(session, "inband_dtmf", NULL,
|
|
inband_dtmf_callback, pvt, 0, SMBF_READ_REPLACE | SMBF_NO_PAUSE | SMBF_ONE_ONLY, &bug)) != SWITCH_STATUS_SUCCESS) {
|
|
return status;
|
|
}
|
|
|
|
switch_channel_set_private(channel, "dtmf", bug);
|
|
|
|
return SWITCH_STATUS_SUCCESS;
|
|
}
|
|
|
|
typedef struct {
|
|
switch_core_session_t *session;
|
|
teletone_generation_session_t ts;
|
|
switch_queue_t *digit_queue;
|
|
switch_buffer_t *audio_buffer;
|
|
switch_mutex_t *mutex;
|
|
int read;
|
|
int ready;
|
|
int skip;
|
|
} switch_inband_dtmf_generate_t;
|
|
|
|
static int teletone_dtmf_generate_handler(teletone_generation_session_t *ts, teletone_tone_map_t *map)
|
|
{
|
|
switch_buffer_t *audio_buffer = ts->user_data;
|
|
int wrote;
|
|
|
|
if (!audio_buffer) {
|
|
return -1;
|
|
}
|
|
|
|
wrote = teletone_mux_tones(ts, map);
|
|
switch_buffer_write(audio_buffer, ts->buffer, wrote * 2);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static switch_status_t generate_on_dtmf(switch_core_session_t *session, const switch_dtmf_t *dtmf, switch_dtmf_direction_t direction)
|
|
{
|
|
switch_channel_t *channel = switch_core_session_get_channel(session);
|
|
switch_media_bug_t *bug = switch_channel_get_private(channel, "dtmf_generate");
|
|
switch_status_t status = SWITCH_STATUS_SUCCESS;
|
|
|
|
if (bug) {
|
|
switch_inband_dtmf_generate_t *pvt = (switch_inband_dtmf_generate_t *) switch_core_media_bug_get_user_data(bug);
|
|
|
|
if (pvt) {
|
|
switch_mutex_lock(pvt->mutex);
|
|
|
|
if (pvt->ready) {
|
|
switch_dtmf_t *dt = NULL;
|
|
switch_zmalloc(dt, sizeof(*dt));
|
|
*dt = *dtmf;
|
|
if (!switch_buffer_inuse(pvt->audio_buffer)) {
|
|
pvt->skip = 10;
|
|
}
|
|
if (switch_queue_trypush(pvt->digit_queue, dt) == SWITCH_STATUS_SUCCESS) {
|
|
switch_event_t *event;
|
|
|
|
if (switch_event_create(&event, SWITCH_EVENT_DTMF) == SWITCH_STATUS_SUCCESS) {
|
|
switch_channel_event_set_data(channel, event);
|
|
switch_event_add_header(event, SWITCH_STACK_BOTTOM, "DTMF-Digit", "%c", dtmf->digit);
|
|
switch_event_add_header(event, SWITCH_STACK_BOTTOM, "DTMF-Duration", "%u", dtmf->duration);
|
|
switch_event_add_header_string(event, SWITCH_STACK_BOTTOM, "DTMF-Source", "APP");
|
|
switch_event_add_header_string(event, SWITCH_STACK_BOTTOM, "DTMF-Conversion", "native:inband");
|
|
if (switch_channel_test_flag(channel, CF_DIVERT_EVENTS)) {
|
|
switch_core_session_queue_event(session, &event);
|
|
} else {
|
|
switch_event_fire(&event);
|
|
}
|
|
}
|
|
|
|
dt = NULL;
|
|
/*
|
|
SWITCH_STATUS_FALSE indicates pretend there never was a DTMF
|
|
since we will be generating it inband now.
|
|
*/
|
|
status = SWITCH_STATUS_FALSE;
|
|
} else {
|
|
free(dt);
|
|
}
|
|
}
|
|
switch_mutex_unlock(pvt->mutex);
|
|
}
|
|
}
|
|
|
|
return status;
|
|
}
|
|
|
|
|
|
static switch_bool_t inband_dtmf_generate_callback(switch_media_bug_t *bug, void *user_data, switch_abc_type_t type)
|
|
{
|
|
switch_inband_dtmf_generate_t *pvt = (switch_inband_dtmf_generate_t *) user_data;
|
|
switch_frame_t *frame;
|
|
switch_codec_implementation_t read_impl = { 0 };
|
|
switch_core_session_get_read_impl(pvt->session, &read_impl);
|
|
|
|
switch (type) {
|
|
case SWITCH_ABC_TYPE_INIT:
|
|
{
|
|
switch_queue_create(&pvt->digit_queue, 100, switch_core_session_get_pool(pvt->session));
|
|
switch_buffer_create_dynamic(&pvt->audio_buffer, 512, 1024, 0);
|
|
teletone_init_session(&pvt->ts, 0, teletone_dtmf_generate_handler, pvt->audio_buffer);
|
|
pvt->ts.rate = read_impl.actual_samples_per_second;
|
|
pvt->ts.channels = 1;
|
|
switch_mutex_init(&pvt->mutex, SWITCH_MUTEX_NESTED, switch_core_session_get_pool(pvt->session));
|
|
if (pvt->read) {
|
|
switch_core_event_hook_add_recv_dtmf(pvt->session, generate_on_dtmf);
|
|
} else {
|
|
switch_core_event_hook_add_send_dtmf(pvt->session, generate_on_dtmf);
|
|
}
|
|
switch_mutex_lock(pvt->mutex);
|
|
pvt->ready = 1;
|
|
switch_mutex_unlock(pvt->mutex);
|
|
}
|
|
break;
|
|
case SWITCH_ABC_TYPE_CLOSE:
|
|
{
|
|
switch_mutex_lock(pvt->mutex);
|
|
pvt->ready = 0;
|
|
switch_core_event_hook_remove_recv_dtmf(pvt->session, generate_on_dtmf);
|
|
switch_buffer_destroy(&pvt->audio_buffer);
|
|
teletone_destroy_session(&pvt->ts);
|
|
switch_mutex_unlock(pvt->mutex);
|
|
}
|
|
break;
|
|
case SWITCH_ABC_TYPE_READ_REPLACE:
|
|
case SWITCH_ABC_TYPE_WRITE_REPLACE:
|
|
{
|
|
switch_size_t bytes;
|
|
void *pop;
|
|
|
|
if (pvt->skip) {
|
|
pvt->skip--;
|
|
return SWITCH_TRUE;
|
|
}
|
|
|
|
|
|
switch_mutex_lock(pvt->mutex);
|
|
|
|
if (!pvt->ready) {
|
|
switch_mutex_unlock(pvt->mutex);
|
|
return SWITCH_FALSE;
|
|
}
|
|
|
|
if (pvt->read) {
|
|
frame = switch_core_media_bug_get_read_replace_frame(bug);
|
|
} else {
|
|
frame = switch_core_media_bug_get_write_replace_frame(bug);
|
|
}
|
|
|
|
if (!switch_buffer_inuse(pvt->audio_buffer)) {
|
|
if (switch_queue_trypop(pvt->digit_queue, &pop) == SWITCH_STATUS_SUCCESS) {
|
|
switch_dtmf_t *dtmf = (switch_dtmf_t *) pop;
|
|
|
|
|
|
if (dtmf->source != SWITCH_DTMF_INBAND_AUDIO) {
|
|
char buf[2] = "";
|
|
int duration = dtmf->duration;
|
|
|
|
buf[0] = dtmf->digit;
|
|
if (duration > (int)switch_core_max_dtmf_duration(0)) {
|
|
duration = switch_core_default_dtmf_duration(0);
|
|
switch_log_printf(SWITCH_CHANNEL_SESSION_LOG(switch_core_media_bug_get_session(bug)),
|
|
SWITCH_LOG_WARNING, "%s Truncating DTMF duration %d ms to %d ms\n",
|
|
switch_channel_get_name(switch_core_session_get_channel(pvt->session)), dtmf->duration / 8, duration);
|
|
}
|
|
|
|
|
|
pvt->ts.duration = duration;
|
|
teletone_run(&pvt->ts, buf);
|
|
}
|
|
free(pop);
|
|
}
|
|
}
|
|
|
|
if (switch_buffer_inuse(pvt->audio_buffer) && (bytes = switch_buffer_read(pvt->audio_buffer, frame->data, frame->datalen))) {
|
|
if (bytes < frame->datalen) {
|
|
switch_byte_t *dp = frame->data;
|
|
memset(dp + bytes, 0, frame->datalen - bytes);
|
|
}
|
|
}
|
|
|
|
if (pvt->read) {
|
|
switch_core_media_bug_set_read_replace_frame(bug, frame);
|
|
} else {
|
|
switch_core_media_bug_set_write_replace_frame(bug, frame);
|
|
}
|
|
|
|
switch_mutex_unlock(pvt->mutex);
|
|
}
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
|
|
return SWITCH_TRUE;
|
|
}
|
|
|
|
SWITCH_DECLARE(switch_status_t) switch_ivr_stop_inband_dtmf_generate_session(switch_core_session_t *session)
|
|
{
|
|
switch_channel_t *channel = switch_core_session_get_channel(session);
|
|
switch_media_bug_t *bug = switch_channel_get_private(channel, "dtmf_generate");
|
|
|
|
if (bug) {
|
|
switch_channel_set_private(channel, "dtmf_generate", NULL);
|
|
switch_core_media_bug_remove(session, &bug);
|
|
return SWITCH_STATUS_SUCCESS;
|
|
}
|
|
|
|
return SWITCH_STATUS_FALSE;
|
|
|
|
}
|
|
|
|
SWITCH_DECLARE(switch_status_t) switch_ivr_inband_dtmf_generate_session(switch_core_session_t *session, switch_bool_t read_stream)
|
|
{
|
|
switch_channel_t *channel = switch_core_session_get_channel(session);
|
|
switch_media_bug_t *bug;
|
|
switch_status_t status;
|
|
switch_inband_dtmf_generate_t *pvt;
|
|
|
|
if ((status = switch_channel_pre_answer(channel)) != SWITCH_STATUS_SUCCESS) {
|
|
return SWITCH_STATUS_FALSE;
|
|
}
|
|
|
|
if (!switch_channel_media_up(channel) || !switch_core_session_get_read_codec(session)) {
|
|
switch_log_printf(SWITCH_CHANNEL_SESSION_LOG(session), SWITCH_LOG_ERROR, "Can not install inband dtmf generate. Media not enabled on channel\n");
|
|
return status;
|
|
}
|
|
|
|
if (!(pvt = switch_core_session_alloc(session, sizeof(*pvt)))) {
|
|
return SWITCH_STATUS_MEMERR;
|
|
}
|
|
|
|
pvt->session = session;
|
|
pvt->read = !!read_stream;
|
|
|
|
if ((status = switch_core_media_bug_add(session, "inband_dtmf_generate", NULL,
|
|
inband_dtmf_generate_callback, pvt, 0,
|
|
SMBF_NO_PAUSE | (pvt->read ? SMBF_READ_REPLACE : SMBF_WRITE_REPLACE) , &bug)) != SWITCH_STATUS_SUCCESS) {
|
|
return status;
|
|
}
|
|
|
|
switch_channel_set_private(channel, "dtmf_generate", bug);
|
|
|
|
return SWITCH_STATUS_SUCCESS;
|
|
}
|
|
|
|
#define MAX_TONES 16
|
|
typedef struct {
|
|
teletone_multi_tone_t mt;
|
|
char *app;
|
|
char *data;
|
|
char *key;
|
|
teletone_tone_map_t map;
|
|
int up;
|
|
int total_hits;
|
|
int hits;
|
|
int sleep;
|
|
int expires;
|
|
int default_sleep;
|
|
int default_expires;
|
|
int once;
|
|
switch_time_t start_time;
|
|
switch_tone_detect_callback_t callback;
|
|
} switch_tone_detect_t;
|
|
|
|
typedef struct {
|
|
switch_tone_detect_t list[MAX_TONES + 1];
|
|
int index;
|
|
switch_media_bug_t *bug;
|
|
switch_core_session_t *session;
|
|
int bug_running;
|
|
int detect_fax;
|
|
} switch_tone_container_t;
|
|
|
|
|
|
static void tone_detect_set_total_time(switch_tone_container_t *cont, int index)
|
|
{
|
|
char *total_time = switch_mprintf("%d", (int)(switch_micro_time_now() - cont->list[index].start_time) / 1000);
|
|
|
|
switch_channel_set_variable_name_printf(switch_core_session_get_channel(cont->session), total_time, "tone_detect_%s_total_time",
|
|
cont->list[index].key);
|
|
switch_safe_free(total_time);
|
|
}
|
|
|
|
static switch_status_t tone_on_dtmf(switch_core_session_t *session, const switch_dtmf_t *dtmf, switch_dtmf_direction_t direction)
|
|
{
|
|
switch_channel_t *channel = switch_core_session_get_channel(session);
|
|
switch_tone_container_t *cont = switch_channel_get_private(channel, "_tone_detect_");
|
|
int i;
|
|
|
|
if (!cont || !cont->detect_fax || dtmf->digit != 'f') {
|
|
return SWITCH_STATUS_SUCCESS;
|
|
}
|
|
|
|
i = cont->detect_fax;
|
|
|
|
tone_detect_set_total_time(cont, i);
|
|
if (cont->list[i].callback) {
|
|
cont->list[i].callback(cont->session, cont->list[i].app, cont->list[i].data);
|
|
} else {
|
|
switch_channel_execute_on(switch_core_session_get_channel(cont->session), SWITCH_CHANNEL_EXECUTE_ON_TONE_DETECT_VARIABLE);
|
|
switch_channel_api_on(switch_core_session_get_channel(cont->session), SWITCH_CHANNEL_API_ON_TONE_DETECT_VARIABLE);
|
|
|
|
if (cont->list[i].app) {
|
|
switch_core_session_execute_application_async(cont->session, cont->list[i].app, cont->list[i].data);
|
|
}
|
|
}
|
|
|
|
return SWITCH_STATUS_SUCCESS;
|
|
|
|
}
|
|
|
|
static switch_bool_t tone_detect_callback(switch_media_bug_t *bug, void *user_data, switch_abc_type_t type)
|
|
{
|
|
switch_tone_container_t *cont = (switch_tone_container_t *) user_data;
|
|
switch_frame_t *frame = NULL;
|
|
int i = 0;
|
|
switch_bool_t rval = SWITCH_TRUE;
|
|
|
|
switch (type) {
|
|
case SWITCH_ABC_TYPE_INIT:
|
|
if (cont) {
|
|
cont->bug_running = 1;
|
|
}
|
|
break;
|
|
case SWITCH_ABC_TYPE_CLOSE:
|
|
break;
|
|
case SWITCH_ABC_TYPE_READ_REPLACE:
|
|
case SWITCH_ABC_TYPE_WRITE_REPLACE:
|
|
{
|
|
|
|
if (type == SWITCH_ABC_TYPE_READ_REPLACE) {
|
|
frame = switch_core_media_bug_get_read_replace_frame(bug);
|
|
} else {
|
|
frame = switch_core_media_bug_get_write_replace_frame(bug);
|
|
}
|
|
|
|
for (i = 0; i < cont->index; i++) {
|
|
int skip = 0;
|
|
|
|
if (cont->list[i].sleep) {
|
|
cont->list[i].sleep--;
|
|
if (cont->list[i].sleep) {
|
|
skip = 1;
|
|
}
|
|
}
|
|
|
|
if (cont->list[i].expires) {
|
|
cont->list[i].expires--;
|
|
if (!cont->list[i].expires) {
|
|
cont->list[i].hits = 0;
|
|
cont->list[i].sleep = 0;
|
|
cont->list[i].expires = 0;
|
|
}
|
|
}
|
|
|
|
if (!cont->list[i].up)
|
|
skip = 1;
|
|
|
|
if (skip)
|
|
continue;
|
|
|
|
if (teletone_multi_tone_detect(&cont->list[i].mt, frame->data, frame->samples)) {
|
|
switch_event_t *event;
|
|
cont->list[i].hits++;
|
|
|
|
switch_log_printf(SWITCH_CHANNEL_SESSION_LOG(switch_core_media_bug_get_session(bug)), SWITCH_LOG_DEBUG, "TONE %s HIT %d/%d\n",
|
|
cont->list[i].key, cont->list[i].hits, cont->list[i].total_hits);
|
|
cont->list[i].sleep = cont->list[i].default_sleep;
|
|
cont->list[i].expires = cont->list[i].default_expires;
|
|
|
|
if (cont->list[i].hits >= cont->list[i].total_hits) {
|
|
switch_log_printf(SWITCH_CHANNEL_SESSION_LOG(switch_core_media_bug_get_session(bug)), SWITCH_LOG_DEBUG, "TONE %s DETECTED\n",
|
|
cont->list[i].key);
|
|
tone_detect_set_total_time(cont, i);
|
|
cont->list[i].up = 0;
|
|
|
|
if (cont->list[i].callback) {
|
|
if ((rval = cont->list[i].callback(cont->session, cont->list[i].app, cont->list[i].data)) == SWITCH_TRUE) {
|
|
switch_log_printf(SWITCH_CHANNEL_SESSION_LOG(switch_core_media_bug_get_session(bug)), SWITCH_LOG_DEBUG, "Re-enabling %s\n",
|
|
cont->list[i].key);
|
|
cont->list[i].up = 1;
|
|
cont->list[i].hits = 0;
|
|
cont->list[i].sleep = 0;
|
|
cont->list[i].expires = 0;
|
|
}
|
|
} else {
|
|
switch_channel_execute_on(switch_core_session_get_channel(cont->session), SWITCH_CHANNEL_EXECUTE_ON_TONE_DETECT_VARIABLE);
|
|
if (cont->list[i].app) {
|
|
switch_core_session_execute_application_async(cont->session, cont->list[i].app, cont->list[i].data);
|
|
}
|
|
}
|
|
|
|
if (cont->list[i].once) {
|
|
rval = SWITCH_FALSE;
|
|
}
|
|
|
|
if (switch_event_create(&event, SWITCH_EVENT_DETECTED_TONE) == SWITCH_STATUS_SUCCESS) {
|
|
switch_event_t *dup;
|
|
switch_event_add_header_string(event, SWITCH_STACK_BOTTOM, "Detected-Tone", cont->list[i].key);
|
|
|
|
if (switch_event_dup(&dup, event) == SWITCH_STATUS_SUCCESS) {
|
|
switch_event_fire(&dup);
|
|
}
|
|
|
|
if (switch_core_session_queue_event(cont->session, &event) != SWITCH_STATUS_SUCCESS) {
|
|
switch_log_printf(SWITCH_CHANNEL_SESSION_LOG(switch_core_media_bug_get_session(bug)), SWITCH_LOG_ERROR,
|
|
"Event queue failed!\n");
|
|
switch_event_add_header_string(event, SWITCH_STACK_BOTTOM, "delivery-failure", "true");
|
|
switch_event_fire(&event);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
break;
|
|
case SWITCH_ABC_TYPE_WRITE:
|
|
default:
|
|
break;
|
|
}
|
|
|
|
if (rval == SWITCH_FALSE) {
|
|
cont->bug_running = 0;
|
|
}
|
|
|
|
return rval;
|
|
}
|
|
|
|
SWITCH_DECLARE(switch_status_t) switch_ivr_stop_tone_detect_session(switch_core_session_t *session)
|
|
{
|
|
switch_channel_t *channel = switch_core_session_get_channel(session);
|
|
switch_tone_container_t *cont = switch_channel_get_private(channel, "_tone_detect_");
|
|
int i = 0;
|
|
|
|
if (cont) {
|
|
switch_channel_set_private(channel, "_tone_detect_", NULL);
|
|
for (i = 0; i < cont->index; i++) {
|
|
cont->list[i].up = 0;
|
|
}
|
|
switch_core_media_bug_remove(session, &cont->bug);
|
|
if (cont->detect_fax) {
|
|
cont->detect_fax = 0;
|
|
}
|
|
return SWITCH_STATUS_SUCCESS;
|
|
}
|
|
return SWITCH_STATUS_FALSE;
|
|
}
|
|
|
|
SWITCH_DECLARE(switch_status_t) switch_ivr_tone_detect_session(switch_core_session_t *session,
|
|
const char *key, const char *tone_spec,
|
|
const char *flags, time_t timeout,
|
|
int hits, const char *app, const char *data, switch_tone_detect_callback_t callback)
|
|
{
|
|
switch_channel_t *channel = switch_core_session_get_channel(session);
|
|
switch_status_t status;
|
|
switch_tone_container_t *cont = switch_channel_get_private(channel, "_tone_detect_");
|
|
char *p, *next;
|
|
int i = 0, ok = 0, detect_fax = 0;
|
|
switch_media_bug_flag_t bflags = 0;
|
|
const char *var;
|
|
switch_codec_implementation_t read_impl = { 0 };
|
|
switch_core_session_get_read_impl(session, &read_impl);
|
|
|
|
|
|
if (zstr(key)) {
|
|
switch_log_printf(SWITCH_CHANNEL_SESSION_LOG(session), SWITCH_LOG_ERROR, "No Key Specified!\n");
|
|
return SWITCH_STATUS_FALSE;
|
|
}
|
|
|
|
if (cont) {
|
|
if (cont->index >= MAX_TONES) {
|
|
switch_log_printf(SWITCH_CHANNEL_SESSION_LOG(session), SWITCH_LOG_ERROR, "Max Tones Reached!\n");
|
|
return SWITCH_STATUS_FALSE;
|
|
}
|
|
|
|
for (i = 0; i < cont->index; i++) {
|
|
if (!zstr(cont->list[i].key) && !strcasecmp(key, cont->list[i].key)) {
|
|
switch_log_printf(SWITCH_CHANNEL_SESSION_LOG(session), SWITCH_LOG_DEBUG, "Re-enabling %s\n", key);
|
|
cont->list[i].up = 1;
|
|
cont->list[i].hits = 0;
|
|
cont->list[i].sleep = 0;
|
|
cont->list[i].expires = 0;
|
|
return SWITCH_STATUS_SUCCESS;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (zstr(tone_spec)) {
|
|
switch_log_printf(SWITCH_CHANNEL_SESSION_LOG(session), SWITCH_LOG_ERROR, "No Spec Specified!\n");
|
|
return SWITCH_STATUS_FALSE;
|
|
}
|
|
|
|
if (!cont && !(cont = switch_core_session_alloc(session, sizeof(*cont)))) {
|
|
return SWITCH_STATUS_MEMERR;
|
|
}
|
|
|
|
if ((var = switch_channel_get_variable(channel, "tone_detect_hits"))) {
|
|
int tmp = atoi(var);
|
|
if (tmp > 0) {
|
|
hits = tmp;
|
|
}
|
|
}
|
|
|
|
if (!hits) hits = 1;
|
|
|
|
switch_log_printf(SWITCH_CHANNEL_SESSION_LOG(session), SWITCH_LOG_DEBUG, "Adding tone spec %s index %d hits %d\n", tone_spec, cont->index, hits);
|
|
|
|
i = 0;
|
|
p = (char *) tone_spec;
|
|
|
|
do {
|
|
teletone_process_t this;
|
|
next = strchr(p, ',');
|
|
while (*p == ' ')
|
|
p++;
|
|
if ((this = (teletone_process_t) atof(p))) {
|
|
ok++;
|
|
cont->list[cont->index].map.freqs[i++] = this;
|
|
}
|
|
if (!strncasecmp(p, "1100", 4)) {
|
|
detect_fax = cont->index;
|
|
}
|
|
|
|
if (next) {
|
|
p = next + 1;
|
|
}
|
|
} while (next);
|
|
cont->list[cont->index].map.freqs[i++] = 0;
|
|
|
|
if (!ok) {
|
|
switch_log_printf(SWITCH_CHANNEL_SESSION_LOG(session), SWITCH_LOG_ERROR, "Invalid tone spec!\n");
|
|
return SWITCH_STATUS_FALSE;
|
|
}
|
|
|
|
cont->detect_fax = detect_fax;
|
|
|
|
cont->list[cont->index].key = switch_core_session_strdup(session, key);
|
|
|
|
if (app) {
|
|
cont->list[cont->index].app = switch_core_session_strdup(session, app);
|
|
}
|
|
|
|
if (data) {
|
|
cont->list[cont->index].data = switch_core_session_strdup(session, data);
|
|
}
|
|
|
|
cont->list[cont->index].callback = callback;
|
|
cont->list[cont->index].hits = 0;
|
|
cont->list[cont->index].total_hits = hits;
|
|
cont->list[cont->index].start_time = switch_micro_time_now();
|
|
|
|
cont->list[cont->index].up = 1;
|
|
memset(&cont->list[cont->index].mt, 0, sizeof(cont->list[cont->index].mt));
|
|
cont->list[cont->index].mt.sample_rate = read_impl.actual_samples_per_second;
|
|
teletone_multi_tone_init(&cont->list[cont->index].mt, &cont->list[cont->index].map);
|
|
cont->session = session;
|
|
|
|
if (switch_channel_pre_answer(channel) != SWITCH_STATUS_SUCCESS) {
|
|
return SWITCH_STATUS_FALSE;
|
|
}
|
|
|
|
cont->list[cont->index].default_sleep = 25;
|
|
cont->list[cont->index].default_expires = 250;
|
|
|
|
if ((var = switch_channel_get_variable(channel, "tone_detect_sleep"))) {
|
|
int tmp = atoi(var);
|
|
if (tmp > 0) {
|
|
cont->list[cont->index].default_sleep = tmp;
|
|
}
|
|
}
|
|
|
|
if ((var = switch_channel_get_variable(channel, "tone_detect_expires"))) {
|
|
int tmp = atoi(var);
|
|
if (tmp > 0) {
|
|
cont->list[cont->index].default_expires = tmp;
|
|
}
|
|
}
|
|
|
|
|
|
if (zstr(flags)) {
|
|
bflags = SMBF_READ_REPLACE;
|
|
} else {
|
|
if (strchr(flags, 'o')) {
|
|
cont->list[cont->index].once = 1;
|
|
}
|
|
|
|
if (strchr(flags, 'r')) {
|
|
bflags |= SMBF_READ_REPLACE;
|
|
} else if (strchr(flags, 'w')) {
|
|
bflags |= SMBF_WRITE_REPLACE;
|
|
}
|
|
}
|
|
|
|
bflags |= SMBF_NO_PAUSE;
|
|
|
|
if (cont->bug_running) {
|
|
switch_log_printf(SWITCH_CHANNEL_SESSION_LOG(session), SWITCH_LOG_DEBUG, "%s bug already running\n", switch_channel_get_name(channel));
|
|
} else {
|
|
cont->bug_running = 1;
|
|
if (cont->detect_fax) {
|
|
switch_core_event_hook_add_send_dtmf(session, tone_on_dtmf);
|
|
switch_core_event_hook_add_recv_dtmf(session, tone_on_dtmf);
|
|
}
|
|
|
|
if ((status = switch_core_media_bug_add(session, "tone_detect", key,
|
|
tone_detect_callback, cont, timeout, bflags, &cont->bug)) != SWITCH_STATUS_SUCCESS) {
|
|
cont->bug_running = 0;
|
|
return status;
|
|
}
|
|
switch_channel_set_private(channel, "_tone_detect_", cont);
|
|
}
|
|
|
|
cont->index++;
|
|
|
|
return SWITCH_STATUS_SUCCESS;
|
|
}
|
|
|
|
|
|
typedef struct {
|
|
const char *app;
|
|
uint32_t flags;
|
|
switch_bind_flag_t bind_flags;
|
|
} dtmf_meta_app_t;
|
|
|
|
typedef struct {
|
|
dtmf_meta_app_t map[14];
|
|
time_t last_digit;
|
|
switch_bool_t meta_on;
|
|
char meta;
|
|
int up;
|
|
} dtmf_meta_settings_t;
|
|
|
|
typedef struct {
|
|
dtmf_meta_settings_t sr[3];
|
|
} dtmf_meta_data_t;
|
|
|
|
#define SWITCH_META_VAR_KEY "__dtmf_meta"
|
|
#define SWITCH_BLOCK_DTMF_KEY "__dtmf_block"
|
|
|
|
typedef struct {
|
|
switch_core_session_t *session;
|
|
const char *app;
|
|
int flags;
|
|
} bch_t;
|
|
|
|
static void *SWITCH_THREAD_FUNC bcast_thread(switch_thread_t *thread, void *obj)
|
|
{
|
|
bch_t *bch = (bch_t *) obj;
|
|
|
|
if (!bch->session) {
|
|
return NULL;
|
|
}
|
|
|
|
if (switch_core_session_read_lock(bch->session) == SWITCH_STATUS_SUCCESS) {
|
|
switch_ivr_broadcast(switch_core_session_get_uuid(bch->session), bch->app, bch->flags);
|
|
switch_core_session_rwunlock(bch->session);
|
|
}
|
|
|
|
return NULL;
|
|
|
|
}
|
|
SWITCH_DECLARE(void) switch_ivr_broadcast_in_thread(switch_core_session_t *session, const char *app, int flags)
|
|
{
|
|
switch_thread_t *thread;
|
|
switch_threadattr_t *thd_attr = NULL;
|
|
switch_memory_pool_t *pool;
|
|
bch_t *bch;
|
|
|
|
switch_assert(session);
|
|
|
|
pool = switch_core_session_get_pool(session);
|
|
|
|
bch = switch_core_session_alloc(session, sizeof(*bch));
|
|
bch->session = session;
|
|
bch->app = app;
|
|
bch->flags = flags;
|
|
|
|
|
|
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, bcast_thread, bch, pool);
|
|
}
|
|
|
|
static switch_status_t meta_on_dtmf(switch_core_session_t *session, const switch_dtmf_t *dtmf, switch_dtmf_direction_t direction)
|
|
{
|
|
switch_channel_t *channel = switch_core_session_get_channel(session);
|
|
dtmf_meta_data_t *md = switch_channel_get_private(channel, SWITCH_META_VAR_KEY);
|
|
time_t now = switch_epoch_time_now(NULL);
|
|
char digit[2] = "";
|
|
int dval;
|
|
|
|
if (!md || switch_channel_test_flag(channel, CF_INNER_BRIDGE)) {
|
|
return SWITCH_STATUS_SUCCESS;
|
|
}
|
|
|
|
if (direction == SWITCH_DTMF_RECV && !md->sr[SWITCH_DTMF_RECV].up) {
|
|
return SWITCH_STATUS_SUCCESS;
|
|
}
|
|
|
|
if (direction == SWITCH_DTMF_SEND && !md->sr[SWITCH_DTMF_SEND].up) {
|
|
return SWITCH_STATUS_SUCCESS;
|
|
}
|
|
|
|
if (md->sr[direction].meta_on && now - md->sr[direction].last_digit > 5) {
|
|
md->sr[direction].meta_on = SWITCH_FALSE;
|
|
switch_log_printf(SWITCH_CHANNEL_SESSION_LOG(session), SWITCH_LOG_ERROR, "%s Meta digit timeout parsing %c\n", switch_channel_get_name(channel),
|
|
dtmf->digit);
|
|
return SWITCH_STATUS_SUCCESS;
|
|
}
|
|
|
|
md->sr[direction].last_digit = now;
|
|
|
|
if (dtmf->digit == md->sr[direction].meta) {
|
|
if (md->sr[direction].meta_on) {
|
|
md->sr[direction].meta_on = SWITCH_FALSE;
|
|
return SWITCH_STATUS_SUCCESS;
|
|
} else {
|
|
md->sr[direction].meta_on = SWITCH_TRUE;
|
|
return SWITCH_STATUS_FALSE;
|
|
}
|
|
}
|
|
|
|
if (md->sr[direction].meta_on) {
|
|
if (is_dtmf(dtmf->digit)) {
|
|
int ok = 0;
|
|
*digit = dtmf->digit;
|
|
dval = switch_dtmftoi(digit);
|
|
|
|
if (direction == SWITCH_DTMF_RECV && (md->sr[direction].map[dval].bind_flags & SBF_DIAL_ALEG)) {
|
|
ok = 1;
|
|
} else if (direction == SWITCH_DTMF_SEND && (md->sr[direction].map[dval].bind_flags & SBF_DIAL_BLEG)) {
|
|
ok = 1;
|
|
}
|
|
|
|
if (ok && md->sr[direction].map[dval].app) {
|
|
uint32_t flags = md->sr[direction].map[dval].flags;
|
|
|
|
if ((md->sr[direction].map[dval].bind_flags & SBF_EXEC_OPPOSITE)) {
|
|
if (direction == SWITCH_DTMF_SEND) {
|
|
flags |= SMF_ECHO_ALEG;
|
|
} else {
|
|
flags |= SMF_ECHO_BLEG;
|
|
}
|
|
} else if ((md->sr[direction].map[dval].bind_flags & SBF_EXEC_SAME)) {
|
|
if (direction == SWITCH_DTMF_SEND) {
|
|
flags |= SMF_ECHO_BLEG;
|
|
} else {
|
|
flags |= SMF_ECHO_ALEG;
|
|
}
|
|
} else if ((md->sr[direction].map[dval].bind_flags & SBF_EXEC_ALEG)) {
|
|
flags |= SMF_ECHO_ALEG;
|
|
} else if ((md->sr[direction].map[dval].bind_flags & SBF_EXEC_BLEG)) {
|
|
flags |= SMF_ECHO_BLEG;
|
|
} else {
|
|
flags |= SMF_ECHO_ALEG;
|
|
}
|
|
|
|
if ((md->sr[direction].map[dval].bind_flags & SBF_EXEC_INLINE)) {
|
|
flags |= SMF_EXEC_INLINE;
|
|
}
|
|
|
|
switch_log_printf(SWITCH_CHANNEL_SESSION_LOG(session), SWITCH_LOG_DEBUG, "%s Processing meta digit '%c' [%s]\n",
|
|
switch_channel_get_name(channel), dtmf->digit, md->sr[direction].map[dval].app);
|
|
|
|
if (switch_channel_test_flag(channel, CF_PROXY_MODE)) {
|
|
switch_ivr_broadcast_in_thread(session, md->sr[direction].map[dval].app, flags | SMF_REBRIDGE);
|
|
} else {
|
|
switch_ivr_broadcast(switch_core_session_get_uuid(session), md->sr[direction].map[dval].app, flags);
|
|
}
|
|
|
|
if ((md->sr[direction].map[dval].bind_flags & SBF_ONCE)) {
|
|
memset(&md->sr[direction].map[dval], 0, sizeof(md->sr[direction].map[dval]));
|
|
switch_log_printf(SWITCH_CHANNEL_SESSION_LOG(session), SWITCH_LOG_DEBUG, "%s Unbinding meta digit '%c'\n",
|
|
switch_channel_get_name(channel), dtmf->digit);
|
|
}
|
|
|
|
} else {
|
|
switch_log_printf(SWITCH_CHANNEL_SESSION_LOG(session), SWITCH_LOG_WARNING, "%s Ignoring meta digit '%c' not mapped\n",
|
|
switch_channel_get_name(channel), dtmf->digit);
|
|
|
|
}
|
|
}
|
|
md->sr[direction].meta_on = SWITCH_FALSE;
|
|
return SWITCH_STATUS_FALSE;
|
|
}
|
|
|
|
return SWITCH_STATUS_SUCCESS;
|
|
}
|
|
|
|
SWITCH_DECLARE(switch_status_t) switch_ivr_unbind_dtmf_meta_session(switch_core_session_t *session, uint32_t key)
|
|
{
|
|
switch_channel_t *channel = switch_core_session_get_channel(session);
|
|
|
|
if (key) {
|
|
dtmf_meta_data_t *md = switch_channel_get_private(channel, SWITCH_META_VAR_KEY);
|
|
|
|
if (!md || key > 9) {
|
|
switch_log_printf(SWITCH_CHANNEL_SESSION_LOG(session), SWITCH_LOG_ERROR, "Invalid key %u\n", key);
|
|
return SWITCH_STATUS_FALSE;
|
|
}
|
|
|
|
memset(&md->sr[SWITCH_DTMF_RECV].map[key], 0, sizeof(md->sr[SWITCH_DTMF_RECV].map[key]));
|
|
memset(&md->sr[SWITCH_DTMF_SEND].map[key], 0, sizeof(md->sr[SWITCH_DTMF_SEND].map[key]));
|
|
switch_log_printf(SWITCH_CHANNEL_SESSION_LOG(session), SWITCH_LOG_INFO, "UnBound A-Leg: %d\n", key);
|
|
|
|
} else {
|
|
switch_log_printf(SWITCH_CHANNEL_SESSION_LOG(session), SWITCH_LOG_INFO, "UnBound A-Leg: ALL\n");
|
|
switch_channel_set_private(channel, SWITCH_META_VAR_KEY, NULL);
|
|
}
|
|
|
|
return SWITCH_STATUS_SUCCESS;
|
|
}
|
|
|
|
static switch_status_t block_on_dtmf(switch_core_session_t *session, const switch_dtmf_t *dtmf, switch_dtmf_direction_t direction)
|
|
{
|
|
switch_channel_t *channel = switch_core_session_get_channel(session);
|
|
uint8_t enabled = (uint8_t)(intptr_t)switch_channel_get_private(channel, SWITCH_BLOCK_DTMF_KEY);
|
|
|
|
if (!enabled || switch_channel_test_flag(channel, CF_INNER_BRIDGE)) {
|
|
return SWITCH_STATUS_SUCCESS;
|
|
}
|
|
|
|
return SWITCH_STATUS_FALSE;
|
|
}
|
|
|
|
SWITCH_DECLARE(switch_status_t) switch_ivr_unblock_dtmf_session(switch_core_session_t *session)
|
|
{
|
|
switch_channel_t *channel = switch_core_session_get_channel(session);
|
|
uint8_t enabled = (uint8_t)(intptr_t)switch_channel_get_private(channel, SWITCH_BLOCK_DTMF_KEY);
|
|
|
|
if (enabled) {
|
|
switch_channel_set_private(channel, SWITCH_BLOCK_DTMF_KEY, NULL);
|
|
}
|
|
|
|
return SWITCH_STATUS_SUCCESS;
|
|
}
|
|
|
|
SWITCH_DECLARE(switch_status_t) switch_ivr_block_dtmf_session(switch_core_session_t *session)
|
|
{
|
|
switch_channel_t *channel = switch_core_session_get_channel(session);
|
|
uint8_t enabled = (uint8_t)(intptr_t)switch_channel_get_private(channel, SWITCH_BLOCK_DTMF_KEY);
|
|
|
|
if (!enabled) {
|
|
switch_channel_set_private(channel, SWITCH_BLOCK_DTMF_KEY, (void *)(intptr_t)1);
|
|
switch_core_event_hook_add_send_dtmf(session, block_on_dtmf);
|
|
switch_core_event_hook_add_recv_dtmf(session, block_on_dtmf);
|
|
}
|
|
|
|
return SWITCH_STATUS_SUCCESS;
|
|
}
|
|
|
|
SWITCH_DECLARE(switch_status_t) switch_ivr_bind_dtmf_meta_session(switch_core_session_t *session, uint32_t key,
|
|
switch_bind_flag_t bind_flags, const char *app)
|
|
{
|
|
switch_channel_t *channel = switch_core_session_get_channel(session);
|
|
dtmf_meta_data_t *md = switch_channel_get_private(channel, SWITCH_META_VAR_KEY);
|
|
const char *meta_var = switch_channel_get_variable(channel, "bind_meta_key");
|
|
char meta = '*';
|
|
char str[2] = "";
|
|
|
|
if (meta_var) {
|
|
char t_meta = *meta_var;
|
|
if (is_dtmf(t_meta)) {
|
|
meta = t_meta;
|
|
} else {
|
|
switch_log_printf(SWITCH_CHANNEL_SESSION_LOG(session), SWITCH_LOG_WARNING, "Invalid META KEY %c\n", t_meta);
|
|
}
|
|
}
|
|
|
|
if (meta != '*' && meta != '#') {
|
|
str[0] = meta;
|
|
|
|
if (switch_dtmftoi(str) == (char)key) {
|
|
switch_log_printf(SWITCH_CHANNEL_SESSION_LOG(session), SWITCH_LOG_ERROR, "Invalid key %u, same as META CHAR\n", key);
|
|
return SWITCH_STATUS_FALSE;
|
|
}
|
|
}
|
|
|
|
|
|
if (key > 13) {
|
|
switch_log_printf(SWITCH_CHANNEL_SESSION_LOG(session), SWITCH_LOG_ERROR, "Invalid key %u\n", key);
|
|
return SWITCH_STATUS_FALSE;
|
|
}
|
|
|
|
if (!md) {
|
|
md = switch_core_session_alloc(session, sizeof(*md));
|
|
switch_channel_set_private(channel, SWITCH_META_VAR_KEY, md);
|
|
switch_core_event_hook_add_send_dtmf(session, meta_on_dtmf);
|
|
switch_core_event_hook_add_recv_dtmf(session, meta_on_dtmf);
|
|
}
|
|
|
|
if (!zstr(app)) {
|
|
if ((bind_flags & SBF_DIAL_ALEG)) {
|
|
md->sr[SWITCH_DTMF_RECV].meta = meta;
|
|
md->sr[SWITCH_DTMF_RECV].up = 1;
|
|
md->sr[SWITCH_DTMF_RECV].map[key].app = switch_core_session_strdup(session, app);
|
|
md->sr[SWITCH_DTMF_RECV].map[key].flags |= SMF_HOLD_BLEG;
|
|
md->sr[SWITCH_DTMF_RECV].map[key].bind_flags = bind_flags;
|
|
|
|
switch_log_printf(SWITCH_CHANNEL_SESSION_LOG(session), SWITCH_LOG_INFO, "Bound A-Leg: %c%c %s\n", meta, switch_itodtmf((char)key), app);
|
|
}
|
|
if ((bind_flags & SBF_DIAL_BLEG)) {
|
|
md->sr[SWITCH_DTMF_SEND].meta = meta;
|
|
md->sr[SWITCH_DTMF_SEND].up = 1;
|
|
md->sr[SWITCH_DTMF_SEND].map[key].app = switch_core_session_strdup(session, app);
|
|
md->sr[SWITCH_DTMF_SEND].map[key].flags |= SMF_HOLD_BLEG;
|
|
md->sr[SWITCH_DTMF_SEND].map[key].bind_flags = bind_flags;
|
|
switch_log_printf(SWITCH_CHANNEL_SESSION_LOG(session), SWITCH_LOG_INFO, "Bound B-Leg: %c%c %s\n", meta, switch_itodtmf((char)key), app);
|
|
}
|
|
|
|
} else {
|
|
if ((bind_flags & SBF_DIAL_ALEG)) {
|
|
switch_log_printf(SWITCH_CHANNEL_SESSION_LOG(session), SWITCH_LOG_INFO, "UnBound A-Leg: %c%c\n", meta, switch_itodtmf((char)key));
|
|
md->sr[SWITCH_DTMF_SEND].map[key].app = NULL;
|
|
} else {
|
|
switch_log_printf(SWITCH_CHANNEL_SESSION_LOG(session), SWITCH_LOG_INFO, "UnBound: B-Leg %c%d\n", meta, key);
|
|
md->sr[SWITCH_DTMF_SEND].map[key].app = NULL;
|
|
}
|
|
}
|
|
|
|
return SWITCH_STATUS_SUCCESS;
|
|
}
|
|
|
|
#define PLAY_AND_DETECT_DONE 1
|
|
#define PLAY_AND_DETECT_DONE_RECOGNIZING 2
|
|
typedef struct {
|
|
int done;
|
|
char *result;
|
|
switch_input_args_t *original_args;
|
|
} play_and_detect_speech_state_t;
|
|
|
|
static void deliver_asr_event(switch_core_session_t *session, switch_event_t *event, switch_input_args_t *args)
|
|
{
|
|
if (args && args->input_callback) {
|
|
args->input_callback(session, (void *)event, SWITCH_INPUT_TYPE_EVENT, args->buf, args->buflen);
|
|
}
|
|
}
|
|
|
|
static switch_status_t play_and_detect_input_callback(switch_core_session_t *session, void *input, switch_input_type_t input_type, void *data, unsigned int len)
|
|
{
|
|
play_and_detect_speech_state_t *state = (play_and_detect_speech_state_t *)data;
|
|
if (!state->done) {
|
|
switch_channel_t *channel = switch_core_session_get_channel(session);
|
|
if (input_type == SWITCH_INPUT_TYPE_EVENT) {
|
|
switch_event_t *event;
|
|
event = (switch_event_t *)input;
|
|
if (event->event_id == SWITCH_EVENT_DETECTED_SPEECH) {
|
|
const char *speech_type = switch_event_get_header(event, "Speech-Type");
|
|
|
|
if (!zstr(speech_type)) {
|
|
deliver_asr_event(session, event, state->original_args);
|
|
|
|
if (!strcasecmp(speech_type, "detected-speech")) {
|
|
const char *result;
|
|
switch_log_printf(SWITCH_CHANNEL_SESSION_LOG(session), SWITCH_LOG_INFO, "(%s) DETECTED SPEECH\n", switch_channel_get_name(channel));
|
|
result = switch_event_get_body(event);
|
|
if (!zstr(result)) {
|
|
state->result = switch_core_session_strdup(session, result);
|
|
} else {
|
|
state->result = "";
|
|
}
|
|
state->original_args = NULL;
|
|
state->done = PLAY_AND_DETECT_DONE_RECOGNIZING;
|
|
return SWITCH_STATUS_BREAK;
|
|
} else if (!strcasecmp(speech_type, "detected-partial-speech")) {
|
|
// ok
|
|
} else if (!strcasecmp(speech_type, "begin-speaking")) {
|
|
switch_log_printf(SWITCH_CHANNEL_SESSION_LOG(session), SWITCH_LOG_INFO, "(%s) START OF SPEECH\n", switch_channel_get_name(channel));
|
|
return SWITCH_STATUS_BREAK;
|
|
} else if (!strcasecmp("closed", speech_type)) {
|
|
state->done = PLAY_AND_DETECT_DONE_RECOGNIZING;
|
|
state->result = "";
|
|
return SWITCH_STATUS_BREAK;
|
|
} else {
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_ERROR, "unhandled speech type %s\n", speech_type);
|
|
}
|
|
}
|
|
}
|
|
} else if (input_type == SWITCH_INPUT_TYPE_DTMF) {
|
|
switch_dtmf_t *dtmf = (switch_dtmf_t *) input;
|
|
const char *terminators = switch_channel_get_variable(channel, SWITCH_PLAYBACK_TERMINATORS_VARIABLE);
|
|
if (terminators) {
|
|
if (!strcasecmp(terminators, "any")) {
|
|
terminators = "1234567890*#";
|
|
} else if (!strcasecmp(terminators, "none")) {
|
|
terminators = NULL;
|
|
}
|
|
}
|
|
if (terminators && strchr(terminators, dtmf->digit)) {
|
|
switch_log_printf(SWITCH_CHANNEL_SESSION_LOG(session), SWITCH_LOG_DEBUG, "(%s) ACCEPT TERMINATOR %c\n", switch_channel_get_name(channel), dtmf->digit);
|
|
switch_channel_set_variable_printf(channel, SWITCH_PLAYBACK_TERMINATOR_USED, "%c", dtmf->digit);
|
|
state->result = switch_core_session_sprintf(session, "DIGIT: %c", dtmf->digit);
|
|
state->done = PLAY_AND_DETECT_DONE;
|
|
return SWITCH_STATUS_BREAK;
|
|
} else {
|
|
switch_log_printf(SWITCH_CHANNEL_SESSION_LOG(session), SWITCH_LOG_DEBUG, "(%s) IGNORE NON-TERMINATOR DIGIT %c\n", switch_channel_get_name(channel), dtmf->digit);
|
|
}
|
|
}
|
|
}
|
|
return SWITCH_STATUS_SUCCESS;
|
|
}
|
|
|
|
SWITCH_DECLARE(switch_status_t) switch_ivr_play_and_detect_speech(switch_core_session_t *session,
|
|
const char *file,
|
|
const char *mod_name,
|
|
const char *grammar,
|
|
char **result,
|
|
uint32_t input_timeout,
|
|
switch_input_args_t *args)
|
|
{
|
|
switch_status_t status = SWITCH_STATUS_FALSE;
|
|
int recognizing = 0;
|
|
switch_input_args_t myargs = { 0 };
|
|
play_and_detect_speech_state_t state = { 0, "", NULL };
|
|
switch_channel_t *channel = switch_core_session_get_channel(session);
|
|
|
|
arg_recursion_check_start(args);
|
|
|
|
if (result == NULL) {
|
|
goto done;
|
|
}
|
|
|
|
if (!input_timeout) input_timeout = 5000;
|
|
|
|
/* start speech detection */
|
|
if ((status = switch_ivr_detect_speech(session, mod_name, grammar, "", NULL, NULL)) != SWITCH_STATUS_SUCCESS) {
|
|
/* map SWITCH_STATUS_FALSE to SWITCH_STATUS_GENERR to indicate grammar load failed
|
|
SWITCH_STATUS_NOT_INITALIZED will be passed back to indicate ASR resource problem */
|
|
if (status == SWITCH_STATUS_FALSE) {
|
|
status = SWITCH_STATUS_GENERR;
|
|
}
|
|
goto done;
|
|
}
|
|
recognizing = 1;
|
|
|
|
/* play the prompt, looking for detection result */
|
|
|
|
if (args) {
|
|
state.original_args = args;
|
|
myargs.dmachine = args->dmachine;
|
|
myargs.read_frame_callback = args->read_frame_callback;
|
|
myargs.user_data = args->user_data;
|
|
}
|
|
|
|
myargs.input_callback = play_and_detect_input_callback;
|
|
myargs.buf = &state;
|
|
myargs.buflen = sizeof(state);
|
|
|
|
status = switch_ivr_play_file(session, NULL, file, &myargs);
|
|
|
|
if (args && args->dmachine && switch_ivr_dmachine_last_ping(args->dmachine) != SWITCH_STATUS_SUCCESS) {
|
|
state.done |= PLAY_AND_DETECT_DONE;
|
|
goto done;
|
|
}
|
|
|
|
if (status != SWITCH_STATUS_BREAK && status != SWITCH_STATUS_SUCCESS) {
|
|
status = SWITCH_STATUS_FALSE;
|
|
goto done;
|
|
}
|
|
|
|
/* wait for result if not done */
|
|
if (!state.done) {
|
|
switch_ivr_detect_speech_start_input_timers(session);
|
|
switch_log_printf(SWITCH_CHANNEL_SESSION_LOG(session), SWITCH_LOG_INFO, "(%s) WAITING FOR RESULT\n", switch_channel_get_name(channel));
|
|
while (!state.done && switch_channel_ready(channel)) {
|
|
status = switch_ivr_sleep(session, input_timeout, SWITCH_FALSE, &myargs);
|
|
|
|
if (args && args->dmachine && switch_ivr_dmachine_last_ping(args->dmachine) != SWITCH_STATUS_SUCCESS) {
|
|
state.done |= PLAY_AND_DETECT_DONE;
|
|
goto done;
|
|
}
|
|
|
|
if (status != SWITCH_STATUS_BREAK && status != SWITCH_STATUS_SUCCESS) {
|
|
status = SWITCH_STATUS_FALSE;
|
|
goto done;
|
|
}
|
|
}
|
|
}
|
|
|
|
done:
|
|
if (recognizing && !(state.done & PLAY_AND_DETECT_DONE_RECOGNIZING)) {
|
|
switch_ivr_pause_detect_speech(session);
|
|
}
|
|
if (recognizing && switch_channel_var_true(channel, "play_and_detect_speech_close_asr")) {
|
|
switch_ivr_stop_detect_speech(session);
|
|
}
|
|
|
|
if (state.done) {
|
|
status = SWITCH_STATUS_SUCCESS;
|
|
}
|
|
if (result) {
|
|
*result = state.result;
|
|
}
|
|
|
|
arg_recursion_check_stop(args);
|
|
|
|
return status;
|
|
}
|
|
|
|
struct speech_thread_handle {
|
|
switch_core_session_t *session;
|
|
switch_asr_handle_t *ah;
|
|
switch_media_bug_t *bug;
|
|
switch_mutex_t *mutex;
|
|
switch_thread_cond_t *cond;
|
|
switch_memory_pool_t *pool;
|
|
switch_thread_t *thread;
|
|
int ready;
|
|
};
|
|
|
|
static void *SWITCH_THREAD_FUNC speech_thread(switch_thread_t *thread, void *obj)
|
|
{
|
|
struct speech_thread_handle *sth = (struct speech_thread_handle *) obj;
|
|
switch_channel_t *channel = switch_core_session_get_channel(sth->session);
|
|
switch_asr_flag_t flags = SWITCH_ASR_FLAG_NONE;
|
|
switch_status_t status;
|
|
switch_event_t *event;
|
|
|
|
switch_thread_cond_create(&sth->cond, sth->pool);
|
|
switch_mutex_init(&sth->mutex, SWITCH_MUTEX_NESTED, sth->pool);
|
|
|
|
if (switch_core_session_read_lock(sth->session) != SWITCH_STATUS_SUCCESS) {
|
|
sth->ready = 0;
|
|
return NULL;
|
|
}
|
|
|
|
switch_mutex_lock(sth->mutex);
|
|
|
|
sth->ready = 1;
|
|
|
|
while (switch_channel_up_nosig(channel) && !switch_test_flag(sth->ah, SWITCH_ASR_FLAG_CLOSED)) {
|
|
char *xmlstr = NULL;
|
|
switch_event_t *headers = NULL;
|
|
|
|
switch_thread_cond_wait(sth->cond, sth->mutex);
|
|
|
|
if (switch_channel_down_nosig(channel) || switch_test_flag(sth->ah, SWITCH_ASR_FLAG_CLOSED)) {
|
|
break;
|
|
}
|
|
|
|
if (switch_core_asr_check_results(sth->ah, &flags) == SWITCH_STATUS_SUCCESS) {
|
|
|
|
status = switch_core_asr_get_results(sth->ah, &xmlstr, &flags);
|
|
|
|
if (status != SWITCH_STATUS_SUCCESS && status != SWITCH_STATUS_BREAK && status != SWITCH_STATUS_MORE_DATA) {
|
|
goto done;
|
|
} else {
|
|
/* Try to fetch extra information for this result, the return value doesn't really matter here - it's just optional data. */
|
|
switch_core_asr_get_result_headers(sth->ah, &headers, &flags);
|
|
}
|
|
|
|
if (status == SWITCH_STATUS_SUCCESS && switch_true(switch_channel_get_variable(channel, "asr_intercept_dtmf"))) {
|
|
const char *p;
|
|
|
|
if ((p = switch_stristr("<input>", xmlstr))) {
|
|
p += 7;
|
|
}
|
|
|
|
while (p && *p) {
|
|
char c;
|
|
|
|
if (*p == '<') {
|
|
break;
|
|
}
|
|
|
|
if (!strncasecmp(p, "pound", 5)) {
|
|
c = '#';
|
|
p += 5;
|
|
} else if (!strncasecmp(p, "hash", 4)) {
|
|
c = '#';
|
|
p += 4;
|
|
} else if (!strncasecmp(p, "star", 4)) {
|
|
c = '*';
|
|
p += 4;
|
|
} else if (!strncasecmp(p, "asterisk", 8)) {
|
|
c = '*';
|
|
p += 8;
|
|
} else {
|
|
c = *p;
|
|
p++;
|
|
}
|
|
|
|
if (is_dtmf(c)) {
|
|
switch_dtmf_t dtmf = {0};
|
|
dtmf.digit = c;
|
|
dtmf.duration = switch_core_default_dtmf_duration(0);
|
|
dtmf.source = SWITCH_DTMF_INBAND_AUDIO;
|
|
switch_log_printf(SWITCH_CHANNEL_CHANNEL_LOG(channel), SWITCH_LOG_DEBUG, "Queue speech detected dtmf %c\n", c);
|
|
switch_channel_queue_dtmf(channel, &dtmf);
|
|
}
|
|
|
|
}
|
|
switch_ivr_resume_detect_speech(sth->session);
|
|
}
|
|
|
|
if (switch_event_create(&event, SWITCH_EVENT_DETECTED_SPEECH) == SWITCH_STATUS_SUCCESS) {
|
|
if (status == SWITCH_STATUS_SUCCESS) {
|
|
switch_event_add_header_string(event, SWITCH_STACK_BOTTOM, "Speech-Type", "detected-speech");
|
|
|
|
if (headers) {
|
|
switch_event_merge(event, headers);
|
|
}
|
|
|
|
switch_event_add_body(event, "%s", xmlstr);
|
|
} else if (status == SWITCH_STATUS_MORE_DATA) {
|
|
switch_event_add_header_string(event, SWITCH_STACK_BOTTOM, "Speech-Type", "detected-partial-speech");
|
|
|
|
if (headers) {
|
|
switch_event_merge(event, headers);
|
|
}
|
|
|
|
switch_event_add_body(event, "%s", xmlstr);
|
|
} else {
|
|
switch_event_add_header_string(event, SWITCH_STACK_BOTTOM, "Speech-Type", "begin-speaking");
|
|
}
|
|
|
|
if (switch_test_flag(sth->ah, SWITCH_ASR_FLAG_FIRE_EVENTS)) {
|
|
switch_event_t *dup;
|
|
|
|
if (switch_event_dup(&dup, event) == SWITCH_STATUS_SUCCESS) {
|
|
switch_channel_event_set_data(channel, dup);
|
|
switch_event_fire(&dup);
|
|
}
|
|
|
|
}
|
|
|
|
if (switch_core_session_queue_event(sth->session, &event) != SWITCH_STATUS_SUCCESS) {
|
|
switch_log_printf(SWITCH_CHANNEL_CHANNEL_LOG(channel), SWITCH_LOG_ERROR, "Event queue failed!\n");
|
|
switch_event_add_header_string(event, SWITCH_STACK_BOTTOM, "delivery-failure", "true");
|
|
switch_event_fire(&event);
|
|
}
|
|
}
|
|
|
|
switch_safe_free(xmlstr);
|
|
|
|
if (headers) {
|
|
switch_event_destroy(&headers);
|
|
}
|
|
}
|
|
}
|
|
done:
|
|
|
|
if (switch_event_create(&event, SWITCH_EVENT_DETECTED_SPEECH) == SWITCH_STATUS_SUCCESS) {
|
|
switch_event_add_header_string(event, SWITCH_STACK_BOTTOM, "Speech-Type", "closed");
|
|
if (switch_test_flag(sth->ah, SWITCH_ASR_FLAG_FIRE_EVENTS)) {
|
|
switch_event_t *dup;
|
|
|
|
if (switch_event_dup(&dup, event) == SWITCH_STATUS_SUCCESS) {
|
|
switch_channel_event_set_data(channel, dup);
|
|
switch_event_fire(&dup);
|
|
}
|
|
|
|
}
|
|
|
|
if (switch_core_session_queue_event(sth->session, &event) != SWITCH_STATUS_SUCCESS) {
|
|
switch_log_printf(SWITCH_CHANNEL_CHANNEL_LOG(channel), SWITCH_LOG_ERROR, "Event queue failed!\n");
|
|
switch_event_add_header_string(event, SWITCH_STACK_BOTTOM, "delivery-failure", "true");
|
|
switch_event_fire(&event);
|
|
}
|
|
}
|
|
|
|
switch_mutex_unlock(sth->mutex);
|
|
switch_core_session_rwunlock(sth->session);
|
|
|
|
return NULL;
|
|
}
|
|
|
|
static switch_bool_t speech_callback(switch_media_bug_t *bug, void *user_data, switch_abc_type_t type)
|
|
{
|
|
struct speech_thread_handle *sth = (struct speech_thread_handle *) user_data;
|
|
uint8_t data[SWITCH_RECOMMENDED_BUFFER_SIZE];
|
|
switch_frame_t frame = { 0 };
|
|
switch_asr_flag_t flags = SWITCH_ASR_FLAG_NONE;
|
|
|
|
frame.data = data;
|
|
frame.buflen = SWITCH_RECOMMENDED_BUFFER_SIZE;
|
|
|
|
switch (type) {
|
|
case SWITCH_ABC_TYPE_INIT:
|
|
{
|
|
switch_threadattr_t *thd_attr = NULL;
|
|
|
|
switch_threadattr_create(&thd_attr, sth->pool);
|
|
switch_threadattr_stacksize_set(thd_attr, SWITCH_THREAD_STACKSIZE);
|
|
switch_thread_create(&sth->thread, thd_attr, speech_thread, sth, sth->pool);
|
|
}
|
|
break;
|
|
case SWITCH_ABC_TYPE_CLOSE:
|
|
{
|
|
switch_status_t st;
|
|
switch_core_session_t *session = switch_core_media_bug_get_session(bug);
|
|
switch_channel_t *channel = switch_core_session_get_channel(session);
|
|
|
|
switch_channel_set_private(channel, SWITCH_SPEECH_KEY, NULL);
|
|
switch_core_event_hook_remove_recv_dtmf(session, speech_on_dtmf);
|
|
|
|
switch_core_asr_close(sth->ah, &flags);
|
|
if (sth->mutex && sth->cond && sth->ready) {
|
|
if (switch_mutex_trylock(sth->mutex) == SWITCH_STATUS_SUCCESS) {
|
|
switch_thread_cond_signal(sth->cond);
|
|
switch_mutex_unlock(sth->mutex);
|
|
}
|
|
}
|
|
|
|
switch_thread_join(&st, sth->thread);
|
|
|
|
}
|
|
break;
|
|
case SWITCH_ABC_TYPE_READ:
|
|
if (sth->ah) {
|
|
if (switch_core_media_bug_read(bug, &frame, SWITCH_FALSE) != SWITCH_STATUS_FALSE) {
|
|
if (switch_core_asr_feed(sth->ah, frame.data, frame.datalen, &flags) != SWITCH_STATUS_SUCCESS) {
|
|
switch_log_printf(SWITCH_CHANNEL_SESSION_LOG(switch_core_media_bug_get_session(bug)), SWITCH_LOG_DEBUG, "Error Feeding Data\n");
|
|
return SWITCH_FALSE;
|
|
}
|
|
if (switch_core_asr_check_results(sth->ah, &flags) == SWITCH_STATUS_SUCCESS) {
|
|
if (sth->mutex && sth->cond && sth->ready) {
|
|
switch_mutex_lock(sth->mutex);
|
|
switch_thread_cond_signal(sth->cond);
|
|
switch_mutex_unlock(sth->mutex);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
break;
|
|
case SWITCH_ABC_TYPE_WRITE:
|
|
default:
|
|
break;
|
|
}
|
|
|
|
return SWITCH_TRUE;
|
|
}
|
|
|
|
static switch_status_t speech_on_dtmf(switch_core_session_t *session, const switch_dtmf_t *dtmf, switch_dtmf_direction_t direction)
|
|
{
|
|
switch_channel_t *channel = switch_core_session_get_channel(session);
|
|
struct speech_thread_handle *sth = switch_channel_get_private(channel, SWITCH_SPEECH_KEY);
|
|
switch_status_t status = SWITCH_STATUS_SUCCESS;
|
|
switch_asr_flag_t flags = SWITCH_ASR_FLAG_NONE;
|
|
|
|
if (sth) {
|
|
if (switch_core_asr_feed_dtmf(sth->ah, dtmf, &flags) != SWITCH_STATUS_SUCCESS) {
|
|
switch_log_printf(SWITCH_CHANNEL_SESSION_LOG(session), SWITCH_LOG_ERROR, "Error Feeding DTMF\n");
|
|
}
|
|
}
|
|
|
|
return status;
|
|
}
|
|
|
|
SWITCH_DECLARE(switch_status_t) switch_ivr_stop_detect_speech(switch_core_session_t *session)
|
|
{
|
|
switch_channel_t *channel = switch_core_session_get_channel(session);
|
|
struct speech_thread_handle *sth;
|
|
|
|
switch_assert(channel != NULL);
|
|
if ((sth = switch_channel_get_private(channel, SWITCH_SPEECH_KEY))) {
|
|
switch_channel_set_private(channel, SWITCH_SPEECH_KEY, NULL);
|
|
switch_core_event_hook_remove_recv_dtmf(session, speech_on_dtmf);
|
|
switch_core_media_bug_remove(session, &sth->bug);
|
|
return SWITCH_STATUS_SUCCESS;
|
|
}
|
|
|
|
return SWITCH_STATUS_FALSE;
|
|
}
|
|
|
|
SWITCH_DECLARE(switch_status_t) switch_ivr_pause_detect_speech(switch_core_session_t *session)
|
|
{
|
|
switch_channel_t *channel = switch_core_session_get_channel(session);
|
|
struct speech_thread_handle *sth = switch_channel_get_private(channel, SWITCH_SPEECH_KEY);
|
|
|
|
if (sth) {
|
|
switch_core_asr_pause(sth->ah);
|
|
return SWITCH_STATUS_SUCCESS;
|
|
}
|
|
return SWITCH_STATUS_FALSE;
|
|
}
|
|
|
|
SWITCH_DECLARE(switch_status_t) switch_ivr_resume_detect_speech(switch_core_session_t *session)
|
|
{
|
|
switch_channel_t *channel = switch_core_session_get_channel(session);
|
|
struct speech_thread_handle *sth = switch_channel_get_private(channel, SWITCH_SPEECH_KEY);
|
|
|
|
if (sth) {
|
|
switch_channel_audio_sync(channel);
|
|
switch_core_asr_resume(sth->ah);
|
|
return SWITCH_STATUS_SUCCESS;
|
|
}
|
|
return SWITCH_STATUS_FALSE;
|
|
}
|
|
|
|
SWITCH_DECLARE(switch_status_t) switch_ivr_detect_speech_load_grammar(switch_core_session_t *session, const char *grammar, const char *name)
|
|
{
|
|
switch_channel_t *channel = switch_core_session_get_channel(session);
|
|
struct speech_thread_handle *sth = switch_channel_get_private(channel, SWITCH_SPEECH_KEY);
|
|
switch_status_t status;
|
|
|
|
if (sth) {
|
|
if ((status = switch_core_asr_load_grammar(sth->ah, grammar, name)) != SWITCH_STATUS_SUCCESS) {
|
|
switch_log_printf(SWITCH_CHANNEL_SESSION_LOG(session), SWITCH_LOG_DEBUG, "Error loading Grammar\n");
|
|
switch_ivr_stop_detect_speech(session);
|
|
}
|
|
return status;
|
|
}
|
|
return SWITCH_STATUS_FALSE;
|
|
}
|
|
|
|
SWITCH_DECLARE(switch_status_t) switch_ivr_set_param_detect_speech(switch_core_session_t *session, const char *name, const char *val)
|
|
{
|
|
struct speech_thread_handle *sth = switch_channel_get_private(switch_core_session_get_channel(session), SWITCH_SPEECH_KEY);
|
|
switch_status_t status = SWITCH_STATUS_FALSE;
|
|
|
|
if (sth && sth->ah && name && val) {
|
|
switch_core_asr_text_param(sth->ah, (char *) name, val);
|
|
status = SWITCH_STATUS_SUCCESS;
|
|
}
|
|
|
|
return status;
|
|
}
|
|
|
|
SWITCH_DECLARE(switch_status_t) switch_ivr_detect_speech_start_input_timers(switch_core_session_t *session)
|
|
{
|
|
switch_channel_t *channel = switch_core_session_get_channel(session);
|
|
struct speech_thread_handle *sth = switch_channel_get_private(channel, SWITCH_SPEECH_KEY);
|
|
|
|
if (sth) {
|
|
switch_core_asr_start_input_timers(sth->ah);
|
|
return SWITCH_STATUS_SUCCESS;
|
|
}
|
|
return SWITCH_STATUS_FALSE;
|
|
}
|
|
|
|
SWITCH_DECLARE(switch_status_t) switch_ivr_detect_speech_unload_grammar(switch_core_session_t *session, const char *name)
|
|
{
|
|
switch_channel_t *channel = switch_core_session_get_channel(session);
|
|
struct speech_thread_handle *sth = switch_channel_get_private(channel, SWITCH_SPEECH_KEY);
|
|
switch_status_t status;
|
|
|
|
if (sth) {
|
|
if ((status = switch_core_asr_unload_grammar(sth->ah, name)) != SWITCH_STATUS_SUCCESS) {
|
|
switch_log_printf(SWITCH_CHANNEL_SESSION_LOG(session), SWITCH_LOG_DEBUG, "Error unloading Grammar\n");
|
|
switch_ivr_stop_detect_speech(session);
|
|
}
|
|
return status;
|
|
}
|
|
return SWITCH_STATUS_FALSE;
|
|
}
|
|
|
|
SWITCH_DECLARE(switch_status_t) switch_ivr_detect_speech_enable_grammar(switch_core_session_t *session, const char *name)
|
|
{
|
|
switch_channel_t *channel = switch_core_session_get_channel(session);
|
|
struct speech_thread_handle *sth = switch_channel_get_private(channel, SWITCH_SPEECH_KEY);
|
|
switch_status_t status;
|
|
|
|
if (sth) {
|
|
if ((status = switch_core_asr_enable_grammar(sth->ah, name)) != SWITCH_STATUS_SUCCESS) {
|
|
switch_log_printf(SWITCH_CHANNEL_SESSION_LOG(session), SWITCH_LOG_DEBUG, "Error enabling Grammar\n");
|
|
switch_ivr_stop_detect_speech(session);
|
|
}
|
|
return status;
|
|
}
|
|
return SWITCH_STATUS_FALSE;
|
|
}
|
|
|
|
SWITCH_DECLARE(switch_status_t) switch_ivr_detect_speech_disable_grammar(switch_core_session_t *session, const char *name)
|
|
{
|
|
switch_channel_t *channel = switch_core_session_get_channel(session);
|
|
struct speech_thread_handle *sth = switch_channel_get_private(channel, SWITCH_SPEECH_KEY);
|
|
switch_status_t status;
|
|
|
|
if (sth) {
|
|
if ((status = switch_core_asr_disable_grammar(sth->ah, name)) != SWITCH_STATUS_SUCCESS) {
|
|
switch_log_printf(SWITCH_CHANNEL_SESSION_LOG(session), SWITCH_LOG_DEBUG, "Error disabling Grammar\n");
|
|
switch_ivr_stop_detect_speech(session);
|
|
}
|
|
return status;
|
|
}
|
|
return SWITCH_STATUS_FALSE;
|
|
}
|
|
|
|
SWITCH_DECLARE(switch_status_t) switch_ivr_detect_speech_disable_all_grammars(switch_core_session_t *session)
|
|
{
|
|
switch_channel_t *channel = switch_core_session_get_channel(session);
|
|
struct speech_thread_handle *sth = switch_channel_get_private(channel, SWITCH_SPEECH_KEY);
|
|
switch_status_t status;
|
|
|
|
if (sth) {
|
|
if ((status = switch_core_asr_disable_all_grammars(sth->ah)) != SWITCH_STATUS_SUCCESS) {
|
|
switch_log_printf(SWITCH_CHANNEL_SESSION_LOG(session), SWITCH_LOG_DEBUG, "Error disabling all Grammars\n");
|
|
switch_ivr_stop_detect_speech(session);
|
|
}
|
|
return status;
|
|
}
|
|
return SWITCH_STATUS_FALSE;
|
|
}
|
|
|
|
SWITCH_DECLARE(switch_status_t) switch_ivr_detect_speech_init(switch_core_session_t *session, const char *mod_name,
|
|
const char *dest, switch_asr_handle_t *ah)
|
|
{
|
|
switch_channel_t *channel = switch_core_session_get_channel(session);
|
|
switch_status_t status;
|
|
switch_asr_flag_t flags = SWITCH_ASR_FLAG_NONE;
|
|
struct speech_thread_handle *sth = switch_channel_get_private(channel, SWITCH_SPEECH_KEY);
|
|
switch_codec_implementation_t read_impl = { 0 };
|
|
const char *p;
|
|
char key[512] = "";
|
|
|
|
if (sth) {
|
|
/* Already initialized */
|
|
return SWITCH_STATUS_SUCCESS;
|
|
}
|
|
|
|
if (!ah) {
|
|
if (!(ah = switch_core_session_alloc(session, sizeof(*ah)))) {
|
|
return SWITCH_STATUS_MEMERR;
|
|
}
|
|
}
|
|
|
|
switch_core_session_get_read_impl(session, &read_impl);
|
|
|
|
if ((status = switch_core_asr_open(ah,
|
|
mod_name,
|
|
"L16",
|
|
read_impl.actual_samples_per_second, dest, &flags,
|
|
switch_core_session_get_pool(session))) != SWITCH_STATUS_SUCCESS) {
|
|
return status;
|
|
}
|
|
|
|
sth = switch_core_session_alloc(session, sizeof(*sth));
|
|
sth->pool = switch_core_session_get_pool(session);
|
|
sth->session = session;
|
|
sth->ah = ah;
|
|
|
|
if ((p = switch_channel_get_variable(channel, "fire_asr_events")) && switch_true(p)) {
|
|
switch_set_flag(ah, SWITCH_ASR_FLAG_FIRE_EVENTS);
|
|
}
|
|
|
|
switch_snprintf(key, sizeof(key), "%s/%s/%s/%s", mod_name, NULL, NULL, dest);
|
|
|
|
if ((status = switch_core_media_bug_add(session, "detect_speech", key,
|
|
speech_callback, sth, 0, SMBF_READ_STREAM | SMBF_NO_PAUSE, &sth->bug)) != SWITCH_STATUS_SUCCESS) {
|
|
switch_core_asr_close(ah, &flags);
|
|
return status;
|
|
}
|
|
|
|
if ((status = switch_core_event_hook_add_recv_dtmf(session, speech_on_dtmf)) != SWITCH_STATUS_SUCCESS) {
|
|
switch_ivr_stop_detect_speech(session);
|
|
return status;
|
|
}
|
|
|
|
switch_channel_set_private(channel, SWITCH_SPEECH_KEY, sth);
|
|
|
|
return SWITCH_STATUS_SUCCESS;
|
|
}
|
|
|
|
static void asr_set_json_text_params(switch_core_session_t *session, switch_asr_handle_t *ah)
|
|
{
|
|
switch_event_header_t *hp;
|
|
switch_event_t *event, *cevent;
|
|
const char *variable_prefix = "asr_json_param_";
|
|
switch_channel_t *channel = switch_core_session_get_channel(session);
|
|
|
|
switch_core_get_variables(&event);
|
|
switch_channel_get_variables(channel, &cevent);
|
|
switch_event_merge(event, cevent);
|
|
|
|
for (hp = event->headers; hp; hp = hp->next) {
|
|
char *var = hp->name;
|
|
char *val = hp->value;
|
|
|
|
if (!strncasecmp(var, variable_prefix, strlen(variable_prefix)) && !zstr(val)) {
|
|
char *json_var = var + strlen(variable_prefix);
|
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_DEBUG, "setting json param %s = %s\n", json_var, val);
|
|
switch_core_asr_text_param(ah, json_var, val);
|
|
}
|
|
}
|
|
|
|
switch_event_destroy(&event);
|
|
switch_event_destroy(&cevent);
|
|
}
|
|
|
|
SWITCH_DECLARE(switch_status_t) switch_ivr_detect_speech(switch_core_session_t *session,
|
|
const char *mod_name,
|
|
const char *grammar, const char *name, const char *dest, switch_asr_handle_t *ah)
|
|
{
|
|
switch_channel_t *channel = switch_core_session_get_channel(session);
|
|
struct speech_thread_handle *sth = switch_channel_get_private(channel, SWITCH_SPEECH_KEY);
|
|
const char *p;
|
|
int resume = 0;
|
|
|
|
|
|
if (!sth) {
|
|
/* No speech thread handle available yet, init speech detection first. */
|
|
if (switch_ivr_detect_speech_init(session, mod_name, dest, ah) != SWITCH_STATUS_SUCCESS) {
|
|
return SWITCH_STATUS_NOT_INITALIZED;
|
|
}
|
|
|
|
/* Fetch the new speech thread handle */
|
|
if (!(sth = switch_channel_get_private(channel, SWITCH_SPEECH_KEY))) {
|
|
return SWITCH_STATUS_NOT_INITALIZED;
|
|
}
|
|
} else {
|
|
resume = 1;
|
|
}
|
|
|
|
asr_set_json_text_params(session, sth->ah);
|
|
|
|
if (switch_core_asr_load_grammar(sth->ah, grammar, name) != SWITCH_STATUS_SUCCESS) {
|
|
switch_log_printf(SWITCH_CHANNEL_SESSION_LOG(session), SWITCH_LOG_DEBUG, "Error loading Grammar\n");
|
|
switch_ivr_stop_detect_speech(session);
|
|
return SWITCH_STATUS_FALSE;
|
|
}
|
|
|
|
if (resume) {
|
|
switch_ivr_resume_detect_speech(session);
|
|
}
|
|
|
|
if ((p = switch_channel_get_variable(channel, "fire_asr_events")) && switch_true(p)) {
|
|
switch_set_flag(sth->ah, SWITCH_ASR_FLAG_FIRE_EVENTS);
|
|
}
|
|
|
|
return SWITCH_STATUS_SUCCESS;
|
|
}
|
|
|
|
struct hangup_helper {
|
|
char uuid_str[SWITCH_UUID_FORMATTED_LENGTH + 1];
|
|
switch_bool_t bleg;
|
|
switch_call_cause_t cause;
|
|
};
|
|
|
|
SWITCH_STANDARD_SCHED_FUNC(sch_hangup_callback)
|
|
{
|
|
struct hangup_helper *helper;
|
|
switch_core_session_t *session, *other_session;
|
|
|
|
switch_assert(task);
|
|
|
|
helper = (struct hangup_helper *) task->cmd_arg;
|
|
|
|
if ((session = switch_core_session_locate(helper->uuid_str))) {
|
|
switch_channel_t *channel = switch_core_session_get_channel(session);
|
|
|
|
if (helper->bleg) {
|
|
if (SWITCH_STATUS_SUCCESS == switch_core_session_get_partner(session, &other_session)) {
|
|
switch_channel_t *other_channel = switch_core_session_get_channel(other_session);
|
|
switch_channel_hangup(other_channel, helper->cause);
|
|
switch_core_session_rwunlock(other_session);
|
|
} else {
|
|
switch_log_printf(SWITCH_CHANNEL_SESSION_LOG(session), SWITCH_LOG_WARNING, "No channel to hangup\n");
|
|
}
|
|
} else {
|
|
switch_channel_hangup(channel, helper->cause);
|
|
}
|
|
|
|
switch_core_session_rwunlock(session);
|
|
}
|
|
}
|
|
|
|
SWITCH_DECLARE(uint32_t) switch_ivr_schedule_hangup(time_t runtime, const char *uuid, switch_call_cause_t cause, switch_bool_t bleg)
|
|
{
|
|
struct hangup_helper *helper;
|
|
size_t len = sizeof(*helper);
|
|
|
|
switch_zmalloc(helper, len);
|
|
|
|
switch_copy_string(helper->uuid_str, uuid, sizeof(helper->uuid_str));
|
|
helper->cause = cause;
|
|
helper->bleg = bleg;
|
|
|
|
return switch_scheduler_add_task(runtime, sch_hangup_callback, (char *) __SWITCH_FUNC__, uuid, 0, helper, SSHF_FREE_ARG);
|
|
}
|
|
|
|
struct transfer_helper {
|
|
char uuid_str[SWITCH_UUID_FORMATTED_LENGTH + 1];
|
|
char *extension;
|
|
char *dialplan;
|
|
char *context;
|
|
};
|
|
|
|
SWITCH_STANDARD_SCHED_FUNC(sch_transfer_callback)
|
|
{
|
|
struct transfer_helper *helper;
|
|
switch_core_session_t *session;
|
|
|
|
switch_assert(task);
|
|
|
|
helper = (struct transfer_helper *) task->cmd_arg;
|
|
|
|
if ((session = switch_core_session_locate(helper->uuid_str))) {
|
|
switch_ivr_session_transfer(session, helper->extension, helper->dialplan, helper->context);
|
|
switch_core_session_rwunlock(session);
|
|
}
|
|
|
|
}
|
|
|
|
SWITCH_DECLARE(uint32_t) switch_ivr_schedule_transfer(time_t runtime, const char *uuid, char *extension, char *dialplan, char *context)
|
|
{
|
|
struct transfer_helper *helper;
|
|
size_t len = sizeof(*helper);
|
|
char *cur = NULL;
|
|
|
|
if (extension) {
|
|
len += strlen(extension) + 1;
|
|
}
|
|
|
|
if (dialplan) {
|
|
len += strlen(dialplan) + 1;
|
|
}
|
|
|
|
if (context) {
|
|
len += strlen(context) + 1;
|
|
}
|
|
|
|
switch_zmalloc(cur, len);
|
|
helper = (struct transfer_helper *) cur;
|
|
|
|
switch_copy_string(helper->uuid_str, uuid, sizeof(helper->uuid_str));
|
|
|
|
cur += sizeof(*helper);
|
|
|
|
if (extension) {
|
|
switch_copy_string(cur, extension, strlen(extension) + 1);
|
|
helper->extension = cur;
|
|
cur += strlen(helper->extension) + 1;
|
|
}
|
|
|
|
if (dialplan) {
|
|
switch_copy_string(cur, dialplan, strlen(dialplan) + 1);
|
|
helper->dialplan = cur;
|
|
cur += strlen(helper->dialplan) + 1;
|
|
}
|
|
|
|
if (context) {
|
|
switch_copy_string(cur, context, strlen(context) + 1);
|
|
helper->context = cur;
|
|
}
|
|
|
|
return switch_scheduler_add_task(runtime, sch_transfer_callback, (char *) __SWITCH_FUNC__, uuid, 0, helper, SSHF_FREE_ARG);
|
|
}
|
|
|
|
struct broadcast_helper {
|
|
char uuid_str[SWITCH_UUID_FORMATTED_LENGTH + 1];
|
|
char *path;
|
|
switch_media_flag_t flags;
|
|
};
|
|
|
|
SWITCH_STANDARD_SCHED_FUNC(sch_broadcast_callback)
|
|
{
|
|
struct broadcast_helper *helper;
|
|
switch_assert(task);
|
|
|
|
helper = (struct broadcast_helper *) task->cmd_arg;
|
|
switch_ivr_broadcast(helper->uuid_str, helper->path, helper->flags);
|
|
}
|
|
|
|
SWITCH_DECLARE(uint32_t) switch_ivr_schedule_broadcast(time_t runtime, const char *uuid, const char *path, switch_media_flag_t flags)
|
|
{
|
|
struct broadcast_helper *helper;
|
|
size_t len = sizeof(*helper) + strlen(path) + 1;
|
|
char *cur = NULL;
|
|
|
|
switch_zmalloc(cur, len);
|
|
helper = (struct broadcast_helper *) cur;
|
|
|
|
cur += sizeof(*helper);
|
|
switch_copy_string(helper->uuid_str, uuid, sizeof(helper->uuid_str));
|
|
helper->flags = flags;
|
|
|
|
switch_copy_string(cur, path, len - sizeof(*helper));
|
|
helper->path = cur;
|
|
|
|
return switch_scheduler_add_task(runtime, sch_broadcast_callback, (char *) __SWITCH_FUNC__, uuid, 0, helper, SSHF_FREE_ARG);
|
|
}
|
|
|
|
SWITCH_DECLARE(switch_status_t) switch_ivr_broadcast(const char *uuid, const char *path, switch_media_flag_t flags)
|
|
{
|
|
switch_channel_t *channel;
|
|
switch_core_session_t *session;
|
|
switch_event_t *event;
|
|
switch_core_session_t *other_session = NULL;
|
|
const char *other_uuid = NULL;
|
|
char *app = "playback";
|
|
char *cause = NULL;
|
|
char *mypath;
|
|
char *p;
|
|
int app_flags = 0, nomedia = 0;
|
|
|
|
switch_assert(path);
|
|
|
|
if (!(session = switch_core_session_locate(uuid))) {
|
|
return SWITCH_STATUS_FALSE;
|
|
}
|
|
|
|
channel = switch_core_session_get_channel(session);
|
|
|
|
mypath = strdup(path);
|
|
switch_assert(mypath);
|
|
|
|
if ((p = strchr(mypath, ':')) && *(p + 1) == ':') {
|
|
app = mypath;
|
|
*p++ = '\0';
|
|
*p++ = '\0';
|
|
path = p;
|
|
}
|
|
|
|
if (switch_channel_test_flag(channel, CF_PROXY_MODE)) {
|
|
nomedia = 1;
|
|
switch_ivr_media(uuid, SMF_REBRIDGE);
|
|
}
|
|
|
|
if ((cause = strchr(app, '!'))) {
|
|
*cause++ = '\0';
|
|
if (!*cause) {
|
|
cause = "normal_clearing";
|
|
}
|
|
}
|
|
|
|
if ((flags & SMF_ECHO_BLEG) && (other_uuid = switch_channel_get_partner_uuid(channel))
|
|
&& (other_session = switch_core_session_locate(other_uuid))) {
|
|
if ((flags & SMF_EXEC_INLINE)) {
|
|
switch_core_session_execute_application_get_flags(other_session, app, path, &app_flags);
|
|
nomedia = 0;
|
|
} else {
|
|
switch_core_session_get_app_flags(app, &app_flags);
|
|
if (switch_event_create(&event, SWITCH_EVENT_COMMAND) == SWITCH_STATUS_SUCCESS) {
|
|
switch_event_add_header_string(event, SWITCH_STACK_BOTTOM, "call-command", "execute");
|
|
switch_event_add_header_string(event, SWITCH_STACK_BOTTOM, "execute-app-name", app);
|
|
switch_event_add_header_string(event, SWITCH_STACK_BOTTOM, "execute-app-arg", path);
|
|
switch_event_add_header_string(event, SWITCH_STACK_BOTTOM, (flags & SMF_PRIORITY) ? "event-lock-pri" : "event-lock", "true");
|
|
|
|
switch_event_add_header(event, SWITCH_STACK_BOTTOM, "lead-frames", "%d", 5);
|
|
|
|
if ((flags & SMF_LOOP)) {
|
|
switch_event_add_header(event, SWITCH_STACK_BOTTOM, "loops", "%d", -1);
|
|
}
|
|
|
|
if ((flags & SMF_HOLD_BLEG)) {
|
|
switch_event_add_header_string(event, SWITCH_STACK_BOTTOM, "hold-bleg", "true");
|
|
}
|
|
|
|
switch_core_session_queue_private_event(other_session, &event, (flags & SMF_PRIORITY));
|
|
}
|
|
}
|
|
|
|
switch_core_session_rwunlock(other_session);
|
|
other_session = NULL;
|
|
}
|
|
|
|
if ((app_flags & SAF_MEDIA_TAP)) {
|
|
nomedia = 0;
|
|
}
|
|
|
|
if ((flags & SMF_ECHO_ALEG)) {
|
|
if ((flags & SMF_EXEC_INLINE)) {
|
|
switch_core_session_execute_application(session, app, path);
|
|
} else {
|
|
if (switch_event_create(&event, SWITCH_EVENT_COMMAND) == SWITCH_STATUS_SUCCESS) {
|
|
switch_event_add_header_string(event, SWITCH_STACK_BOTTOM, "call-command", "execute");
|
|
switch_event_add_header_string(event, SWITCH_STACK_BOTTOM, "execute-app-name", app);
|
|
switch_event_add_header_string(event, SWITCH_STACK_BOTTOM, "execute-app-arg", path);
|
|
switch_event_add_header_string(event, SWITCH_STACK_BOTTOM, (flags & SMF_PRIORITY) ? "event-lock-pri" : "event-lock", "true");
|
|
switch_event_add_header(event, SWITCH_STACK_BOTTOM, "lead-frames", "%d", 5);
|
|
|
|
if ((flags & SMF_LOOP)) {
|
|
switch_event_add_header(event, SWITCH_STACK_BOTTOM, "loops", "%d", -1);
|
|
}
|
|
if ((flags & SMF_HOLD_BLEG)) {
|
|
switch_event_add_header_string(event, SWITCH_STACK_BOTTOM, "hold-bleg", "true");
|
|
}
|
|
|
|
switch_core_session_queue_private_event(session, &event, (flags & SMF_PRIORITY));
|
|
|
|
if (nomedia)
|
|
switch_channel_set_flag(channel, CF_BROADCAST_DROP_MEDIA);
|
|
}
|
|
}
|
|
}
|
|
|
|
if (cause) {
|
|
if (switch_event_create(&event, SWITCH_EVENT_COMMAND) == SWITCH_STATUS_SUCCESS) {
|
|
switch_event_add_header_string(event, SWITCH_STACK_BOTTOM, "call-command", "execute");
|
|
switch_event_add_header_string(event, SWITCH_STACK_BOTTOM, "execute-app-name", "hangup");
|
|
switch_event_add_header_string(event, SWITCH_STACK_BOTTOM, "execute-app-arg", cause);
|
|
switch_event_add_header_string(event, SWITCH_STACK_BOTTOM, (flags & SMF_PRIORITY) ? "event-lock-pri" : "event-lock", "true");
|
|
switch_core_session_queue_private_event(session, &event, (flags & SMF_PRIORITY));
|
|
}
|
|
}
|
|
|
|
switch_core_session_rwunlock(session);
|
|
switch_safe_free(mypath);
|
|
|
|
return SWITCH_STATUS_SUCCESS;
|
|
}
|
|
|
|
|
|
typedef struct oht_s {
|
|
switch_image_t *img;
|
|
switch_img_position_t pos;
|
|
uint8_t alpha;
|
|
} overly_helper_t;
|
|
|
|
static switch_bool_t video_write_overlay_callback(switch_media_bug_t *bug, void *user_data, switch_abc_type_t type)
|
|
{
|
|
overly_helper_t *oht = (overly_helper_t *) user_data;
|
|
switch_core_session_t *session = switch_core_media_bug_get_session(bug);
|
|
switch_channel_t *channel = switch_core_session_get_channel(session);
|
|
|
|
switch (type) {
|
|
case SWITCH_ABC_TYPE_INIT:
|
|
{
|
|
}
|
|
break;
|
|
case SWITCH_ABC_TYPE_CLOSE:
|
|
{
|
|
switch_img_free(&oht->img);
|
|
}
|
|
break;
|
|
case SWITCH_ABC_TYPE_WRITE_VIDEO_PING:
|
|
if (switch_channel_test_flag(channel, CF_VIDEO_DECODED_READ)) {
|
|
switch_frame_t *frame = switch_core_media_bug_get_video_ping_frame(bug);
|
|
int x = 0, y = 0;
|
|
switch_image_t *oimg = NULL;
|
|
|
|
if (frame->img && oht->img) {
|
|
switch_img_copy(oht->img, &oimg);
|
|
switch_img_fit(&oimg, frame->img->d_w, frame->img->d_h, SWITCH_FIT_SIZE);
|
|
switch_img_find_position(oht->pos, frame->img->d_w, frame->img->d_h, oimg->d_w, oimg->d_h, &x, &y);
|
|
switch_img_overlay(frame->img, oimg, x, y, oht->alpha);
|
|
//switch_img_patch(frame->img, oimg, x, y);
|
|
switch_img_free(&oimg);
|
|
}
|
|
}
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
|
|
return SWITCH_TRUE;
|
|
}
|
|
|
|
|
|
SWITCH_DECLARE(switch_status_t) switch_ivr_stop_video_write_overlay_session(switch_core_session_t *session)
|
|
{
|
|
switch_channel_t *channel = switch_core_session_get_channel(session);
|
|
switch_media_bug_t *bug = switch_channel_get_private(channel, "_video_write_overlay_bug_");
|
|
|
|
if (bug) {
|
|
switch_channel_set_private(channel, "_video_write_overlay_bug_", NULL);
|
|
switch_core_media_bug_remove(session, &bug);
|
|
return SWITCH_STATUS_SUCCESS;
|
|
}
|
|
|
|
return SWITCH_STATUS_FALSE;
|
|
}
|
|
|
|
SWITCH_DECLARE(switch_status_t) switch_ivr_video_write_overlay_session(switch_core_session_t *session, const char *img_path,
|
|
switch_img_position_t pos, uint8_t alpha)
|
|
{
|
|
switch_channel_t *channel = switch_core_session_get_channel(session);
|
|
switch_status_t status;
|
|
switch_media_bug_flag_t bflags = SMBF_WRITE_VIDEO_PING;
|
|
switch_media_bug_t *bug;
|
|
overly_helper_t *oht;
|
|
switch_image_t *img;
|
|
|
|
bflags |= SMBF_NO_PAUSE;
|
|
|
|
if (switch_channel_get_private(channel, "_video_write_overlay_bug_")) {
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_ERROR, "Only one of this type of bug per channel\n");
|
|
return SWITCH_STATUS_FALSE;
|
|
}
|
|
|
|
if (!(img = switch_img_read_png(img_path, SWITCH_IMG_FMT_ARGB))) {
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_ERROR, "Error opening file: %s\n", img_path);
|
|
return SWITCH_STATUS_FALSE;
|
|
}
|
|
|
|
oht = switch_core_session_alloc(session, sizeof(*oht));
|
|
oht->img = img;
|
|
oht->pos = pos;
|
|
oht->alpha = alpha;
|
|
|
|
if ((status = switch_core_media_bug_add(session, "video_write_overlay", NULL,
|
|
video_write_overlay_callback, oht, 0, bflags, &bug)) != SWITCH_STATUS_SUCCESS) {
|
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_ERROR, "Error creating bug, file: %s\n", img_path);
|
|
switch_img_free(&oht->img);
|
|
return status;
|
|
}
|
|
|
|
switch_channel_set_private(channel, "_video_write_overlay_bug_", bug);
|
|
|
|
return SWITCH_STATUS_SUCCESS;
|
|
}
|
|
|
|
/* For Emacs:
|
|
* Local Variables:
|
|
* mode:c
|
|
* indent-tabs-mode:t
|
|
* tab-width:4
|
|
* c-basic-offset:4
|
|
* End:
|
|
* For VIM:
|
|
* vim:set softtabstop=4 shiftwidth=4 tabstop=4 noet:
|
|
*/
|