mirror of
https://github.com/asterisk/asterisk.git
synced 2025-10-24 05:38:11 +00:00
Merge in-access updates for queue priorities (bug #1821)
git-svn-id: https://origsvn.digium.com/svn/asterisk/trunk@3288 65c4cc65-6c06-0410-ace0-fbb531ad65f3
This commit is contained in:
143
apps/app_queue.c
143
apps/app_queue.c
@@ -7,6 +7,8 @@
|
|||||||
*
|
*
|
||||||
* Mark Spencer <markster@linux-support.net>
|
* Mark Spencer <markster@linux-support.net>
|
||||||
*
|
*
|
||||||
|
* 2004-06-04: Priorities in queues added by inAccess Networks (work funded by Hellas On Line (HOL) www.hol.gr).
|
||||||
|
*
|
||||||
* These features added by David C. Troy <dave@toad.net>:
|
* These features added by David C. Troy <dave@toad.net>:
|
||||||
* - Per-queue holdtime calculation
|
* - Per-queue holdtime calculation
|
||||||
* - Estimated holdtime announcement
|
* - Estimated holdtime announcement
|
||||||
@@ -155,6 +157,7 @@ struct queue_ent {
|
|||||||
char announce[80]; /* Announcement to play for member when call is answered */
|
char announce[80]; /* Announcement to play for member when call is answered */
|
||||||
char context[80]; /* Context when user exits queue */
|
char context[80]; /* Context when user exits queue */
|
||||||
int pos; /* Where we are in the queue */
|
int pos; /* Where we are in the queue */
|
||||||
|
int prio; /* Our priority */
|
||||||
int last_pos_said; /* Last position we told the user */
|
int last_pos_said; /* Last position we told the user */
|
||||||
time_t last_pos; /* Last time we told the user their position */
|
time_t last_pos; /* Last time we told the user their position */
|
||||||
int opos; /* Where we started in the queue */
|
int opos; /* Where we started in the queue */
|
||||||
@@ -169,7 +172,7 @@ struct member {
|
|||||||
char tech[80]; /* Technology */
|
char tech[80]; /* Technology */
|
||||||
char loc[256]; /* Location */
|
char loc[256]; /* Location */
|
||||||
int penalty; /* Are we a last resort? */
|
int penalty; /* Are we a last resort? */
|
||||||
int calls;
|
int calls; /* Number of calls serviced by this member */
|
||||||
int dynamic; /* Are we dynamically added? */
|
int dynamic; /* Are we dynamically added? */
|
||||||
time_t lastcall; /* When last successful call was hungup */
|
time_t lastcall; /* When last successful call was hungup */
|
||||||
struct member *next; /* Next member */
|
struct member *next; /* Next member */
|
||||||
@@ -238,12 +241,35 @@ static int strat2int(char *strategy)
|
|||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* Insert the 'new' entry after the 'prev' entry of queue 'q' */
|
||||||
|
static inline void insert_entry(struct ast_call_queue *q,
|
||||||
|
struct queue_ent *prev, struct queue_ent *new, int *pos)
|
||||||
|
{
|
||||||
|
struct queue_ent *cur;
|
||||||
|
|
||||||
|
if (!q || !new)
|
||||||
|
return;
|
||||||
|
if (prev) {
|
||||||
|
cur = prev->next;
|
||||||
|
prev->next = new;
|
||||||
|
} else {
|
||||||
|
cur = q->head;
|
||||||
|
q->head = new;
|
||||||
|
}
|
||||||
|
new->next = cur;
|
||||||
|
new->parent = q;
|
||||||
|
new->pos = ++(*pos);
|
||||||
|
new->opos = *pos;
|
||||||
|
}
|
||||||
|
|
||||||
static int join_queue(char *queuename, struct queue_ent *qe)
|
static int join_queue(char *queuename, struct queue_ent *qe)
|
||||||
{
|
{
|
||||||
struct ast_call_queue *q;
|
struct ast_call_queue *q;
|
||||||
struct queue_ent *cur, *prev = NULL;
|
struct queue_ent *cur, *prev = NULL;
|
||||||
int res = -1;
|
int res = -1;
|
||||||
int pos = 0;
|
int pos = 0;
|
||||||
|
int inserted = 0;
|
||||||
|
|
||||||
ast_mutex_lock(&qlock);
|
ast_mutex_lock(&qlock);
|
||||||
q = queues;
|
q = queues;
|
||||||
while(q) {
|
while(q) {
|
||||||
@@ -251,24 +277,27 @@ static int join_queue(char *queuename, struct queue_ent *qe)
|
|||||||
/* This is our one */
|
/* This is our one */
|
||||||
ast_mutex_lock(&q->lock);
|
ast_mutex_lock(&q->lock);
|
||||||
if (q->members && (!q->maxlen || (q->count < q->maxlen))) {
|
if (q->members && (!q->maxlen || (q->count < q->maxlen))) {
|
||||||
/* There's space for us, put us at the end */
|
/* There's space for us, put us at the right position inside
|
||||||
|
* the queue.
|
||||||
|
* Take into account the priority of the calling user */
|
||||||
|
inserted = 0;
|
||||||
prev = NULL;
|
prev = NULL;
|
||||||
cur = q->head;
|
cur = q->head;
|
||||||
while(cur) {
|
while(cur) {
|
||||||
|
/* We have higher priority than the current user, enter
|
||||||
|
* before him, after all the other users with priority
|
||||||
|
* higher or equal to our priority. */
|
||||||
|
if ((!inserted) && (qe->prio > cur->prio)) {
|
||||||
|
insert_entry(q, prev, qe, &pos);
|
||||||
|
inserted = 1;
|
||||||
|
}
|
||||||
cur->pos = ++pos;
|
cur->pos = ++pos;
|
||||||
prev = cur;
|
prev = cur;
|
||||||
cur = cur->next;
|
cur = cur->next;
|
||||||
}
|
}
|
||||||
if (prev)
|
/* No luck, join at the end of the queue */
|
||||||
prev->next = qe;
|
if (!inserted)
|
||||||
else
|
insert_entry(q, prev, qe, &pos);
|
||||||
q->head = qe;
|
|
||||||
/* Fix additional pointers and
|
|
||||||
information */
|
|
||||||
qe->next = NULL;
|
|
||||||
qe->parent = q;
|
|
||||||
qe->pos = ++pos;
|
|
||||||
qe->opos = pos;
|
|
||||||
strncpy(qe->moh, q->moh, sizeof(qe->moh));
|
strncpy(qe->moh, q->moh, sizeof(qe->moh));
|
||||||
strncpy(qe->announce, q->announce, sizeof(qe->announce));
|
strncpy(qe->announce, q->announce, sizeof(qe->announce));
|
||||||
strncpy(qe->context, q->context, sizeof(qe->context));
|
strncpy(qe->context, q->context, sizeof(qe->context));
|
||||||
@@ -565,12 +594,15 @@ static int ring_one(struct queue_ent *qe, struct localuser *outgoing)
|
|||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
/* Ring just the best channel */
|
/* Ring just the best channel */
|
||||||
ast_log(LOG_DEBUG, "Trying '%s/%s' with metric %d\n", best->tech, best->numsubst, best->metric);
|
if (option_debug)
|
||||||
|
ast_log(LOG_DEBUG, "Trying '%s/%s' with metric %d\n",
|
||||||
|
best->tech, best->numsubst, best->metric);
|
||||||
ring_entry(qe, best);
|
ring_entry(qe, best);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
} while (best && !best->chan);
|
} while (best && !best->chan);
|
||||||
if (!best) {
|
if (!best) {
|
||||||
|
if (option_debug)
|
||||||
ast_log(LOG_DEBUG, "Nobody left to try ringing in queue\n");
|
ast_log(LOG_DEBUG, "Nobody left to try ringing in queue\n");
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
@@ -784,6 +816,26 @@ static struct localuser *wait_for_answer(struct queue_ent *qe, struct localuser
|
|||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static int is_our_turn(struct queue_ent *qe)
|
||||||
|
{
|
||||||
|
struct queue_ent *ch;
|
||||||
|
int res;
|
||||||
|
|
||||||
|
/* Atomically read the parent head -- does not need a lock */
|
||||||
|
ch = qe->parent->head;
|
||||||
|
/* If we are now at the top of the head, break out */
|
||||||
|
if (ch == qe) {
|
||||||
|
if (option_debug)
|
||||||
|
ast_log(LOG_DEBUG, "It's our turn (%s).\n", qe->chan->name);
|
||||||
|
res = 1;
|
||||||
|
} else {
|
||||||
|
if (option_debug)
|
||||||
|
ast_log(LOG_DEBUG, "It's not our turn (%s).\n", qe->chan->name);
|
||||||
|
res = 0;
|
||||||
|
}
|
||||||
|
return res;
|
||||||
|
}
|
||||||
|
|
||||||
static int wait_our_turn(struct queue_ent *qe, int ringing)
|
static int wait_our_turn(struct queue_ent *qe, int ringing)
|
||||||
{
|
{
|
||||||
struct queue_ent *ch;
|
struct queue_ent *ch;
|
||||||
@@ -796,8 +848,11 @@ static int wait_our_turn(struct queue_ent *qe, int ringing)
|
|||||||
ch = qe->parent->head;
|
ch = qe->parent->head;
|
||||||
|
|
||||||
/* If we are now at the top of the head, break out */
|
/* If we are now at the top of the head, break out */
|
||||||
if (qe == ch)
|
if (ch == qe) {
|
||||||
|
if (option_debug)
|
||||||
|
ast_log(LOG_DEBUG, "It's our turn (%s).\n", qe->chan->name);
|
||||||
break;
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
/* If we have timed out, break out */
|
/* If we have timed out, break out */
|
||||||
if ( qe->queuetimeout ) {
|
if ( qe->queuetimeout ) {
|
||||||
@@ -810,7 +865,6 @@ static int wait_our_turn(struct queue_ent *qe, int ringing)
|
|||||||
if (qe->parent->announcefrequency && !ringing)
|
if (qe->parent->announcefrequency && !ringing)
|
||||||
say_position(qe);
|
say_position(qe);
|
||||||
|
|
||||||
|
|
||||||
/* Wait a second before checking again */
|
/* Wait a second before checking again */
|
||||||
res = ast_waitfordigit(qe->chan, RECHECK * 1000);
|
res = ast_waitfordigit(qe->chan, RECHECK * 1000);
|
||||||
if (res)
|
if (res)
|
||||||
@@ -919,6 +973,9 @@ static int try_calling(struct queue_ent *qe, char *options, char *announceoverri
|
|||||||
struct ast_bridge_config config;
|
struct ast_bridge_config config;
|
||||||
/* Hold the lock while we setup the outgoing calls */
|
/* Hold the lock while we setup the outgoing calls */
|
||||||
ast_mutex_lock(&qe->parent->lock);
|
ast_mutex_lock(&qe->parent->lock);
|
||||||
|
if (option_debug)
|
||||||
|
ast_log(LOG_DEBUG, "%s is trying to call a queue member.\n",
|
||||||
|
qe->chan->name);
|
||||||
strncpy(queuename, qe->parent->name, sizeof(queuename) - 1);
|
strncpy(queuename, qe->parent->name, sizeof(queuename) - 1);
|
||||||
time(&now);
|
time(&now);
|
||||||
cur = qe->parent->members;
|
cur = qe->parent->members;
|
||||||
@@ -952,10 +1009,12 @@ static int try_calling(struct queue_ent *qe, char *options, char *announceoverri
|
|||||||
if ((strchr(options, 'n')) && (now - qe->start >= qe->parent->timeout))
|
if ((strchr(options, 'n')) && (now - qe->start >= qe->parent->timeout))
|
||||||
*go_on = 1;
|
*go_on = 1;
|
||||||
}
|
}
|
||||||
if (url) {
|
if (option_debug) {
|
||||||
|
if (url)
|
||||||
ast_log(LOG_DEBUG, "Queue with URL=%s_\n", url);
|
ast_log(LOG_DEBUG, "Queue with URL=%s_\n", url);
|
||||||
} else
|
else
|
||||||
ast_log(LOG_DEBUG, "Simple queue (no URL)\n");
|
ast_log(LOG_DEBUG, "Simple queue (no URL)\n");
|
||||||
|
}
|
||||||
|
|
||||||
tmp->member = cur; /* Never directly dereference! Could change on reload */
|
tmp->member = cur; /* Never directly dereference! Could change on reload */
|
||||||
strncpy(tmp->tech, cur->tech, sizeof(tmp->tech)-1);
|
strncpy(tmp->tech, cur->tech, sizeof(tmp->tech)-1);
|
||||||
@@ -1009,6 +1068,8 @@ static int try_calling(struct queue_ent *qe, char *options, char *announceoverri
|
|||||||
/* Nobody answered, next please? */
|
/* Nobody answered, next please? */
|
||||||
res=0;
|
res=0;
|
||||||
}
|
}
|
||||||
|
if (option_debug)
|
||||||
|
ast_log(LOG_DEBUG, "%s: Nobody answered.\n", qe->chan->name);
|
||||||
goto out;
|
goto out;
|
||||||
}
|
}
|
||||||
if (peer) {
|
if (peer) {
|
||||||
@@ -1078,6 +1139,7 @@ static int try_calling(struct queue_ent *qe, char *options, char *announceoverri
|
|||||||
/* Drop out of the queue at this point, to prepare for next caller */
|
/* Drop out of the queue at this point, to prepare for next caller */
|
||||||
leave_queue(qe);
|
leave_queue(qe);
|
||||||
if( url && !ast_strlen_zero(url) && ast_channel_supports_html(peer) ) {
|
if( url && !ast_strlen_zero(url) && ast_channel_supports_html(peer) ) {
|
||||||
|
if (option_debug)
|
||||||
ast_log(LOG_DEBUG, "app_queue: sendurl=%s.\n", url);
|
ast_log(LOG_DEBUG, "app_queue: sendurl=%s.\n", url);
|
||||||
ast_channel_sendurl( peer, url );
|
ast_channel_sendurl( peer, url );
|
||||||
}
|
}
|
||||||
@@ -1382,12 +1444,13 @@ static int queue_exec(struct ast_channel *chan, void *data)
|
|||||||
char *options = NULL;
|
char *options = NULL;
|
||||||
char *url = NULL;
|
char *url = NULL;
|
||||||
char *announceoverride = NULL;
|
char *announceoverride = NULL;
|
||||||
|
char *user_priority;
|
||||||
|
int prio;
|
||||||
char *queuetimeoutstr = NULL;
|
char *queuetimeoutstr = NULL;
|
||||||
|
|
||||||
/* whether to exit Queue application after the timeout hits */
|
/* whether to exit Queue application after the timeout hits */
|
||||||
int go_on = 0;
|
int go_on = 0;
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
/* Our queue entry */
|
/* Our queue entry */
|
||||||
struct queue_ent qe;
|
struct queue_ent qe;
|
||||||
|
|
||||||
@@ -1430,23 +1493,43 @@ static int queue_exec(struct ast_channel *chan, void *data)
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* Get the priority from the variable ${QUEUE_PRIO} */
|
||||||
|
user_priority = pbx_builtin_getvar_helper(chan, "QUEUE_PRIO");
|
||||||
|
if (user_priority) {
|
||||||
|
if (sscanf(user_priority, "%d", &prio) == 1) {
|
||||||
|
if (option_debug)
|
||||||
|
ast_log(LOG_DEBUG, "%s: Got priority %d from ${QUEUE_PRIO}.\n",
|
||||||
|
chan->name, prio);
|
||||||
|
} else {
|
||||||
|
ast_log(LOG_WARNING, "${QUEUE_PRIO}: Invalid value (%s), channel %s.\n",
|
||||||
|
user_priority, chan->name);
|
||||||
|
prio = 0;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
if (option_debug)
|
||||||
|
ast_log(LOG_DEBUG, "NO QUEUE_PRIO variable found. Using default.\n");
|
||||||
|
prio = 0;
|
||||||
|
}
|
||||||
|
|
||||||
if (options) {
|
if (options) {
|
||||||
if (strchr(options, 'r')) {
|
if (strchr(options, 'r')) {
|
||||||
ringing = 1;
|
ringing = 1;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
printf("queue: %s, options: %s, url: %s, announce: %s, timeout: %d\n",
|
// if (option_debug)
|
||||||
queuename, options, url, announceoverride, qe.queuetimeout);
|
ast_log(LOG_DEBUG, "queue: %s, options: %s, url: %s, announce: %s, timeout: %d, priority: %d\n",
|
||||||
|
queuename, options, url, announceoverride, qe.queuetimeout, (int)prio);
|
||||||
|
|
||||||
qe.chan = chan;
|
qe.chan = chan;
|
||||||
qe.start = time(NULL);
|
qe.start = time(NULL);
|
||||||
|
qe.prio = (int)prio;
|
||||||
qe.last_pos_said = 0;
|
qe.last_pos_said = 0;
|
||||||
qe.last_pos = 0;
|
qe.last_pos = 0;
|
||||||
if (!join_queue(queuename, &qe)) {
|
if (!join_queue(queuename, &qe)) {
|
||||||
ast_queue_log(queuename, chan->uniqueid, "NONE", "ENTERQUEUE", "%s|%s", url ? url : "", chan->callerid ? chan->callerid : "");
|
ast_queue_log(queuename, chan->uniqueid, "NONE", "ENTERQUEUE", "%s|%s", url ? url : "", chan->callerid ? chan->callerid : "");
|
||||||
/* Start music on hold */
|
/* Start music on hold */
|
||||||
|
check_turns:
|
||||||
if (ringing) {
|
if (ringing) {
|
||||||
ast_indicate(chan, AST_CONTROL_RINGING);
|
ast_indicate(chan, AST_CONTROL_RINGING);
|
||||||
} else {
|
} else {
|
||||||
@@ -1532,7 +1615,15 @@ static int queue_exec(struct ast_channel *chan, void *data)
|
|||||||
res = 0;
|
res = 0;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
/* Since this is a priority queue and
|
||||||
|
* it is not sure that we are still at the head
|
||||||
|
* of the queue, go and check for our turn again.
|
||||||
|
*/
|
||||||
|
if (!is_our_turn(&qe)) {
|
||||||
|
ast_log(LOG_DEBUG, "Darn priorities, going back in queue (%s)!\n",
|
||||||
|
qe.chan->name);
|
||||||
|
goto check_turns;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
/* Don't allow return code > 0 */
|
/* Don't allow return code > 0 */
|
||||||
@@ -1733,7 +1824,7 @@ static void reload_queues(void)
|
|||||||
if (!q->count) {
|
if (!q->count) {
|
||||||
free(q);
|
free(q);
|
||||||
} else
|
} else
|
||||||
ast_log(LOG_WARNING, "XXX Leaking a litttle memory :( XXX\n");
|
ast_log(LOG_WARNING, "XXX Leaking a little memory :( XXX\n");
|
||||||
} else
|
} else
|
||||||
ql = q;
|
ql = q;
|
||||||
q = qn;
|
q = qn;
|
||||||
@@ -1809,8 +1900,8 @@ static int __queues_show(int fd, int argc, char **argv, int queue_show)
|
|||||||
pos = 1;
|
pos = 1;
|
||||||
ast_cli(fd, " Callers: \n");
|
ast_cli(fd, " Callers: \n");
|
||||||
for (qe = q->head; qe; qe = qe->next)
|
for (qe = q->head; qe; qe = qe->next)
|
||||||
ast_cli(fd, " %d. %s (wait: %ld:%2.2ld)\n", pos++, qe->chan->name,
|
ast_cli(fd, " %d. %s (wait: %ld:%2.2ld, prio: %d)\n", pos++, qe->chan->name,
|
||||||
(long)(now - qe->start) / 60, (long)(now - qe->start) % 60);
|
(long)(now - qe->start) / 60, (long)(now - qe->start) % 60, qe->prio);
|
||||||
} else
|
} else
|
||||||
ast_cli(fd, " No Callers\n");
|
ast_cli(fd, " No Callers\n");
|
||||||
ast_cli(fd, "\n");
|
ast_cli(fd, "\n");
|
||||||
|
Reference in New Issue
Block a user