app_queue: Fix multiple calls to a queue member that is in only one queue.

When ringinuse=no queue members can receive more than one call if these
calls happen at nearly the same time.

* Fix so a queue member does not receive more than one call from a queue.

NOTE: This fix does not prevent multiple calls to a member if the member
is in more than one queue.

* Did some refactoring to eliminate some code redundancy.

(issue ASTERISK-16115)
Reported by: nik600
Patches:
      jira_asterisk_16115_single_q_v1.8.patch (license #5621) patch uploaded by rmudgett
      Modified


git-svn-id: https://origsvn.digium.com/svn/asterisk/branches/1.8@378663 65c4cc65-6c06-0410-ace0-fbb531ad65f3
changes/98/198/1
Richard Mudgett 13 years ago
parent 6a94f73341
commit 7ba4fe48ca

@ -1046,6 +1046,7 @@ struct member {
struct call_queue *lastqueue; /*!< Last queue we received a call */ struct call_queue *lastqueue; /*!< Last queue we received a call */
unsigned int dead:1; /*!< Used to detect members deleted in realtime */ unsigned int dead:1; /*!< Used to detect members deleted in realtime */
unsigned int delme:1; /*!< Flag to delete entry on reload */ unsigned int delme:1; /*!< Flag to delete entry on reload */
unsigned int call_pending:1; /*!< TRUE if the Q is attempting to place a call to the member. */
char rt_uniqueid[80]; /*!< Unique id of realtime member entry */ char rt_uniqueid[80]; /*!< Unique id of realtime member entry */
}; };
@ -3079,6 +3080,112 @@ static char *vars2manager(struct ast_channel *chan, char *vars, size_t len)
return vars; return vars;
} }
/*!
* \internal
* \brief Check if the member status is available.
*
* \param status Member status to check if available.
*
* \retval non-zero if the member status is available.
*/
static int member_status_available(int status)
{
return status == AST_DEVICE_NOT_INUSE || status == AST_DEVICE_UNKNOWN;
}
/*!
* \internal
* \brief Clear the member call pending flag.
*
* \param mem Queue member.
*
* \return Nothing
*/
static void member_call_pending_clear(struct member *mem)
{
ao2_lock(mem);
mem->call_pending = 0;
ao2_unlock(mem);
}
/*!
* \internal
* \brief Set the member call pending flag.
*
* \param mem Queue member.
*
* \retval non-zero if call pending flag was already set.
*/
static int member_call_pending_set(struct member *mem)
{
int old_pending;
ao2_lock(mem);
old_pending = mem->call_pending;
mem->call_pending = 1;
ao2_unlock(mem);
return old_pending;
}
/*!
* \internal
* \brief Determine if can ring a queue entry.
*
* \param qe Queue entry to check.
* \param call Member call attempt.
*
* \retval non-zero if an entry can be called.
*/
static int can_ring_entry(struct queue_ent *qe, struct callattempt *call)
{
if (call->member->paused) {
ast_debug(1, "%s paused, can't receive call\n", call->interface);
return 0;
}
if (!qe->parent->ringinuse && !member_status_available(call->member->status)) {
ast_debug(1, "%s not available, can't receive call\n", call->interface);
return 0;
}
if ((call->lastqueue && call->lastqueue->wrapuptime && (time(NULL) - call->lastcall < call->lastqueue->wrapuptime))
|| (!call->lastqueue && qe->parent->wrapuptime && (time(NULL) - call->lastcall < qe->parent->wrapuptime))) {
ast_debug(1, "Wrapuptime not yet expired on queue %s for %s\n",
(call->lastqueue ? call->lastqueue->name : qe->parent->name),
call->interface);
return 0;
}
if (use_weight && compare_weight(qe->parent, call->member)) {
ast_debug(1, "Priority queue delaying call to %s:%s\n",
qe->parent->name, call->interface);
return 0;
}
if (!qe->parent->ringinuse) {
if (member_call_pending_set(call->member)) {
ast_debug(1, "%s has another call pending, can't receive call\n",
call->interface);
return 0;
}
/*
* The queue member is available. Get current status to be sure
* because the device state and extension state callbacks may
* not have updated the status yet.
*/
if (!member_status_available(get_queue_member_status(call->member))) {
ast_debug(1, "%s actually not available, can't receive call\n",
call->interface);
member_call_pending_clear(call->member);
return 0;
}
}
return 1;
}
/*! /*!
* \brief Part 2 of ring_one * \brief Part 2 of ring_one
* *
@ -3102,42 +3209,15 @@ static int ring_entry(struct queue_ent *qe, struct callattempt *tmp, int *busies
const char *macrocontext, *macroexten; const char *macrocontext, *macroexten;
/* on entry here, we know that tmp->chan == NULL */ /* on entry here, we know that tmp->chan == NULL */
if ((tmp->lastqueue && tmp->lastqueue->wrapuptime && (time(NULL) - tmp->lastcall < tmp->lastqueue->wrapuptime)) || if (!can_ring_entry(qe, tmp)) {
(!tmp->lastqueue && qe->parent->wrapuptime && (time(NULL) - tmp->lastcall < qe->parent->wrapuptime))) { if (qe->chan->cdr) {
ast_debug(1, "Wrapuptime not yet expired on queue %s for %s\n",
(tmp->lastqueue ? tmp->lastqueue->name : qe->parent->name), tmp->interface);
if (qe->chan->cdr)
ast_cdr_busy(qe->chan->cdr);
tmp->stillgoing = 0;
(*busies)++;
return 0;
}
if (!qe->parent->ringinuse && (tmp->member->status != AST_DEVICE_NOT_INUSE) && (tmp->member->status != AST_DEVICE_UNKNOWN)) {
ast_debug(1, "%s in use, can't receive call\n", tmp->interface);
if (qe->chan->cdr)
ast_cdr_busy(qe->chan->cdr);
tmp->stillgoing = 0;
(*busies)++;
return 0;
}
if (tmp->member->paused) {
ast_debug(1, "%s paused, can't receive call\n", tmp->interface);
if (qe->chan->cdr)
ast_cdr_busy(qe->chan->cdr); ast_cdr_busy(qe->chan->cdr);
tmp->stillgoing = 0;
(*busies)++;
return 0;
} }
if (use_weight && compare_weight(qe->parent,tmp->member)) {
ast_debug(1, "Priority queue delaying call to %s:%s\n", qe->parent->name, tmp->interface);
if (qe->chan->cdr)
ast_cdr_busy(qe->chan->cdr);
tmp->stillgoing = 0; tmp->stillgoing = 0;
(*busies)++; ++*busies;
return 0; return 0;
} }
ast_assert(qe->parent->ringinuse || tmp->member->call_pending);
ast_copy_string(tech, tmp->interface, sizeof(tech)); ast_copy_string(tech, tmp->interface, sizeof(tech));
if ((location = strchr(tech, '/'))) if ((location = strchr(tech, '/')))
@ -3148,17 +3228,18 @@ static int ring_entry(struct queue_ent *qe, struct callattempt *tmp, int *busies
/* Request the peer */ /* Request the peer */
tmp->chan = ast_request(tech, qe->chan->nativeformats, qe->chan, location, &status); tmp->chan = ast_request(tech, qe->chan->nativeformats, qe->chan, location, &status);
if (!tmp->chan) { /* If we can't, just go on to the next call */ if (!tmp->chan) { /* If we can't, just go on to the next call */
if (qe->chan->cdr)
ast_cdr_busy(qe->chan->cdr);
tmp->stillgoing = 0;
ao2_lock(qe->parent); ao2_lock(qe->parent);
update_status(qe->parent, tmp->member, get_queue_member_status(tmp->member));
qe->parent->rrpos++; qe->parent->rrpos++;
qe->linpos++; qe->linpos++;
ao2_unlock(qe->parent); ao2_unlock(qe->parent);
(*busies)++; member_call_pending_clear(tmp->member);
if (qe->chan->cdr) {
ast_cdr_busy(qe->chan->cdr);
}
tmp->stillgoing = 0;
++*busies;
return 0; return 0;
} }
@ -3233,10 +3314,12 @@ static int ring_entry(struct queue_ent *qe, struct callattempt *tmp, int *busies
/* Again, keep going even if there's an error */ /* Again, keep going even if there's an error */
ast_verb(3, "Couldn't call %s\n", tmp->interface); ast_verb(3, "Couldn't call %s\n", tmp->interface);
do_hang(tmp); do_hang(tmp);
(*busies)++; member_call_pending_clear(tmp->member);
update_status(qe->parent, tmp->member, get_queue_member_status(tmp->member)); ++*busies;
return 0; return 0;
} else if (qe->parent->eventwhencalled) { }
if (qe->parent->eventwhencalled) {
char vars[2048]; char vars[2048];
ast_channel_lock_both(tmp->chan, qe->chan); ast_channel_lock_both(tmp->chan, qe->chan);
@ -3270,7 +3353,7 @@ static int ring_entry(struct queue_ent *qe, struct callattempt *tmp, int *busies
ast_verb(3, "Called %s\n", tmp->interface); ast_verb(3, "Called %s\n", tmp->interface);
} }
update_status(qe->parent, tmp->member, get_queue_member_status(tmp->member)); member_call_pending_clear(tmp->member);
return 1; return 1;
} }
@ -7472,7 +7555,7 @@ static int manager_queues_summary(struct mansession *s, const struct message *m)
while ((mem = ao2_iterator_next(&mem_iter))) { while ((mem = ao2_iterator_next(&mem_iter))) {
if ((mem->status != AST_DEVICE_UNAVAILABLE) && (mem->status != AST_DEVICE_INVALID)) { if ((mem->status != AST_DEVICE_UNAVAILABLE) && (mem->status != AST_DEVICE_INVALID)) {
++qmemcount; ++qmemcount;
if (((mem->status == AST_DEVICE_NOT_INUSE) || (mem->status == AST_DEVICE_UNKNOWN)) && !(mem->paused)) { if (member_status_available(mem->status) && !mem->paused) {
++qmemavail; ++qmemavail;
} }
} }

Loading…
Cancel
Save