FS-7045 #resolve #comment [mod_rayo] guarantee that dialed call can be joined when answered event is sent.
This commit is contained in:
parent
c327455cde
commit
e8cf9c7d35
|
@ -141,6 +141,10 @@ struct rayo_call {
|
||||||
switch_event_t *end_event;
|
switch_event_t *end_event;
|
||||||
/** True if ringing event sent to client */
|
/** True if ringing event sent to client */
|
||||||
int ringing_sent;
|
int ringing_sent;
|
||||||
|
/** true if rayo app has started */
|
||||||
|
int rayo_app_started;
|
||||||
|
/** delayed delivery of answer event because rayo APP wasn't started yet */
|
||||||
|
switch_event_t *answer_event;
|
||||||
/** True if request to create this call failed */
|
/** True if request to create this call failed */
|
||||||
int dial_request_failed;
|
int dial_request_failed;
|
||||||
};
|
};
|
||||||
|
@ -1167,6 +1171,9 @@ done:
|
||||||
if (event) {
|
if (event) {
|
||||||
switch_event_destroy(&event);
|
switch_event_destroy(&event);
|
||||||
}
|
}
|
||||||
|
if (call->answer_event) {
|
||||||
|
switch_event_destroy(&call->answer_event);
|
||||||
|
}
|
||||||
switch_core_hash_destroy(&call->pcps);
|
switch_core_hash_destroy(&call->pcps);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1350,6 +1357,8 @@ static struct rayo_call *rayo_call_init(struct rayo_call *call, switch_memory_po
|
||||||
call->dial_request_id = NULL;
|
call->dial_request_id = NULL;
|
||||||
call->end_event = NULL;
|
call->end_event = NULL;
|
||||||
call->dial_request_failed = 0;
|
call->dial_request_failed = 0;
|
||||||
|
call->rayo_app_started = 0;
|
||||||
|
call->answer_event = NULL;
|
||||||
switch_core_hash_init(&call->pcps);
|
switch_core_hash_init(&call->pcps);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -2080,31 +2089,38 @@ static iks *join_call(struct rayo_call *call, switch_core_session_t *session, st
|
||||||
|
|
||||||
/* check if joining to rayo call */
|
/* check if joining to rayo call */
|
||||||
struct rayo_call *b_call = RAYO_CALL_LOCATE(call_uri);
|
struct rayo_call *b_call = RAYO_CALL_LOCATE(call_uri);
|
||||||
if (!b_call) {
|
if (b_call) {
|
||||||
/* not a rayo call */
|
if (!call->rayo_app_started) {
|
||||||
response = iks_new_error_detailed(node, STANZA_ERROR_SERVICE_UNAVAILABLE, "b-leg is gone");
|
/* A-leg not under rayo control yet */
|
||||||
} else if (!has_call_control(b_call, msg)) {
|
response = iks_new_error_detailed(node, STANZA_ERROR_UNEXPECTED_REQUEST, "a-leg is not ready to join");
|
||||||
/* not allowed to join to this call */
|
} else if (!b_call->rayo_app_started) {
|
||||||
response = iks_new_error(node, STANZA_ERROR_NOT_ALLOWED);
|
/* B-leg not under rayo control yet */
|
||||||
} else if (b_call->joined) {
|
response = iks_new_error_detailed(node, STANZA_ERROR_UNEXPECTED_REQUEST, "b-leg is not ready to join");
|
||||||
/* don't support multiple joined calls */
|
} else if (!has_call_control(b_call, msg)) {
|
||||||
response = iks_new_error_detailed(node, STANZA_ERROR_CONFLICT, "multiple joined calls not supported");
|
/* not allowed to join to this call */
|
||||||
|
response = iks_new_error(node, STANZA_ERROR_NOT_ALLOWED);
|
||||||
|
} else if (b_call->joined) {
|
||||||
|
/* don't support multiple joined calls */
|
||||||
|
response = iks_new_error_detailed(node, STANZA_ERROR_CONFLICT, "multiple joined calls not supported");
|
||||||
|
} else {
|
||||||
|
/* bridge this call to call-uri */
|
||||||
|
switch_channel_set_variable(switch_core_session_get_channel(session), "bypass_media", bypass);
|
||||||
|
if (switch_false(bypass)) {
|
||||||
|
switch_channel_pre_answer(switch_core_session_get_channel(session));
|
||||||
|
}
|
||||||
|
call->pending_join_request = iks_copy(node);
|
||||||
|
if (switch_ivr_uuid_bridge(rayo_call_get_uuid(call), rayo_call_get_uuid(b_call)) != SWITCH_STATUS_SUCCESS) {
|
||||||
|
iks *request = call->pending_join_request;
|
||||||
|
iks *result = iks_new_error(request, STANZA_ERROR_SERVICE_UNAVAILABLE);
|
||||||
|
call->pending_join_request = NULL;
|
||||||
|
RAYO_SEND_REPLY(call, iks_find_attrib_soft(request, "from"), result);
|
||||||
|
iks_delete(call->pending_join_request);
|
||||||
|
}
|
||||||
|
}
|
||||||
RAYO_RELEASE(b_call);
|
RAYO_RELEASE(b_call);
|
||||||
} else {
|
} else {
|
||||||
/* bridge this call to call-uri */
|
/* not a rayo call */
|
||||||
switch_channel_set_variable(switch_core_session_get_channel(session), "bypass_media", bypass);
|
response = iks_new_error_detailed(node, STANZA_ERROR_SERVICE_UNAVAILABLE, "b-leg is gone");
|
||||||
if (switch_false(bypass)) {
|
|
||||||
switch_channel_pre_answer(switch_core_session_get_channel(session));
|
|
||||||
}
|
|
||||||
call->pending_join_request = iks_copy(node);
|
|
||||||
if (switch_ivr_uuid_bridge(rayo_call_get_uuid(call), rayo_call_get_uuid(b_call)) != SWITCH_STATUS_SUCCESS) {
|
|
||||||
iks *request = call->pending_join_request;
|
|
||||||
iks *result = iks_new_error(request, STANZA_ERROR_SERVICE_UNAVAILABLE);
|
|
||||||
call->pending_join_request = NULL;
|
|
||||||
RAYO_SEND_REPLY(call, iks_find_attrib_soft(request, "from"), result);
|
|
||||||
iks_delete(call->pending_join_request);
|
|
||||||
}
|
|
||||||
RAYO_RELEASE(b_call);
|
|
||||||
}
|
}
|
||||||
return response;
|
return response;
|
||||||
}
|
}
|
||||||
|
@ -2187,7 +2203,10 @@ static iks *join_mixer(struct rayo_call *call, switch_core_session_t *session, s
|
||||||
iks *node = msg->payload;
|
iks *node = msg->payload;
|
||||||
iks *response = NULL;
|
iks *response = NULL;
|
||||||
|
|
||||||
if (call->joined_id) {
|
if (!call->rayo_app_started) {
|
||||||
|
/* A-leg not under rayo control yet */
|
||||||
|
response = iks_new_error_detailed(node, STANZA_ERROR_UNEXPECTED_REQUEST, "call is not ready to join");
|
||||||
|
} else if (call->joined_id) {
|
||||||
/* adjust join conference params */
|
/* adjust join conference params */
|
||||||
if (!strcmp("duplex", direction)) {
|
if (!strcmp("duplex", direction)) {
|
||||||
if ((response = exec_conference_api(session, mixer_name, "unmute", node)) ||
|
if ((response = exec_conference_api(session, mixer_name, "unmute", node)) ||
|
||||||
|
@ -3379,10 +3398,17 @@ static void on_call_answer_event(struct rayo_client *rclient, switch_event_t *ev
|
||||||
{
|
{
|
||||||
struct rayo_call *call = RAYO_CALL_LOCATE_BY_ID(switch_event_get_header(event, "Unique-ID"));
|
struct rayo_call *call = RAYO_CALL_LOCATE_BY_ID(switch_event_get_header(event, "Unique-ID"));
|
||||||
if (call) {
|
if (call) {
|
||||||
iks *revent = iks_new_presence("answered", RAYO_NS,
|
switch_mutex_lock(RAYO_ACTOR(call)->mutex);
|
||||||
switch_event_get_header(event, "variable_rayo_call_jid"),
|
if (call->rayo_app_started) {
|
||||||
switch_event_get_header(event, "variable_rayo_dcp_jid"));
|
iks *revent = iks_new_presence("answered", RAYO_NS,
|
||||||
RAYO_SEND_MESSAGE(call, RAYO_JID(rclient), revent);
|
switch_event_get_header(event, "variable_rayo_call_jid"),
|
||||||
|
switch_event_get_header(event, "variable_rayo_dcp_jid"));
|
||||||
|
RAYO_SEND_MESSAGE(call, RAYO_JID(rclient), revent);
|
||||||
|
} else if (!call->answer_event) {
|
||||||
|
/* delay sending this event until the rayo APP has started */
|
||||||
|
switch_event_dup(&call->answer_event, event);
|
||||||
|
}
|
||||||
|
switch_mutex_unlock(RAYO_ACTOR(call)->mutex);
|
||||||
RAYO_RELEASE(call);
|
RAYO_RELEASE(call);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -3840,6 +3866,13 @@ SWITCH_STANDARD_APP(rayo_app)
|
||||||
clients_to_offer_count = switch_separate_string(data_dup, ',', clients_to_offer, sizeof(clients_to_offer) / sizeof(clients_to_offer[0]));
|
clients_to_offer_count = switch_separate_string(data_dup, ',', clients_to_offer, sizeof(clients_to_offer) / sizeof(clients_to_offer[0]));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* It is now safe for inbound call to be fully controlled by rayo client */
|
||||||
|
if (switch_channel_direction(channel) == SWITCH_CALL_DIRECTION_INBOUND) {
|
||||||
|
switch_mutex_lock(RAYO_ACTOR(call)->mutex);
|
||||||
|
call->rayo_app_started = 1;
|
||||||
|
switch_mutex_unlock(RAYO_ACTOR(call)->mutex);
|
||||||
|
}
|
||||||
|
|
||||||
/* Offer call to all (or specified) ONLINE clients */
|
/* Offer call to all (or specified) ONLINE clients */
|
||||||
/* TODO load balance offers so first session doesn't always get offer first? */
|
/* TODO load balance offers so first session doesn't always get offer first? */
|
||||||
switch_mutex_lock(globals.clients_mutex);
|
switch_mutex_lock(globals.clients_mutex);
|
||||||
|
@ -3880,11 +3913,31 @@ done:
|
||||||
switch_channel_set_variable(channel, "hold_hangup_xfer_exten", "park:inline:");
|
switch_channel_set_variable(channel, "hold_hangup_xfer_exten", "park:inline:");
|
||||||
switch_channel_set_variable(channel, SWITCH_SEND_SILENCE_WHEN_IDLE_VARIABLE, "-1"); /* required so that output mixing works */
|
switch_channel_set_variable(channel, SWITCH_SEND_SILENCE_WHEN_IDLE_VARIABLE, "-1"); /* required so that output mixing works */
|
||||||
switch_core_event_hook_add_read_frame(session, rayo_call_on_read_frame);
|
switch_core_event_hook_add_read_frame(session, rayo_call_on_read_frame);
|
||||||
|
|
||||||
if (switch_channel_direction(channel) == SWITCH_CALL_DIRECTION_OUTBOUND) {
|
if (switch_channel_direction(channel) == SWITCH_CALL_DIRECTION_OUTBOUND) {
|
||||||
|
/* At this point, this outbound call might already be under control of a rayo client that is waiting for answer before sending
|
||||||
|
commands. The answered event might have been sent before we are ready to execute commands, so we delayed sending
|
||||||
|
those events if the rayo APP hadn't started yet. This delay would have only been a few milliseconds.
|
||||||
|
*/
|
||||||
|
switch_mutex_lock(RAYO_ACTOR(call)->mutex);
|
||||||
|
call->rayo_app_started = 1;
|
||||||
|
if (call->answer_event) {
|
||||||
|
struct rayo_client *rclient = RAYO_CLIENT(RAYO_LOCATE(rayo_call_get_dcp_jid(call)));
|
||||||
|
if (rclient) {
|
||||||
|
on_call_answer_event(rclient, call->answer_event);
|
||||||
|
switch_event_destroy(&call->answer_event);
|
||||||
|
RAYO_RELEASE(rclient);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
switch_mutex_unlock(RAYO_ACTOR(call)->mutex);
|
||||||
|
|
||||||
|
/* Outbound calls might have a nested join to another call or conference - do that now */
|
||||||
if (!zstr(app)) {
|
if (!zstr(app)) {
|
||||||
switch_core_session_execute_application(session, app, app_args);
|
switch_core_session_execute_application(session, app, app_args);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* Ready for remote control */
|
||||||
switch_ivr_park(session, NULL);
|
switch_ivr_park(session, NULL);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue