mirror of
https://github.com/asterisk/asterisk.git
synced 2025-09-04 11:58:52 +00:00
app_queue.c: Fix setting QUEUE_MEMBER 'paused' and 'ringinuse'.
Setting the 'paused' and 'ringinuse' options on a queue member using the dialplan function QUEUE_MEMBER did not behave the same way as the equivalent dialplan applications or AMI actions. * Made queue_function_mem_write() call the set_member_paused() and set_member_value() for the 'paused' and 'ringinuse' options respectively. A beneficial side effect is that the queue name is now optional and sets the value in all queues the interface is a member. * Update QUEUE_MEMBER XML documentation. * Fix error checking in QUEUE_MEMBER() write. ASTERISK-25215 #close Reported by: Lorne Gaetz Change-Id: I3a016be8dc94d63a9cc155295ff9c9afa5f707cb
This commit is contained in:
@@ -549,7 +549,7 @@ ASTERISK_REGISTER_FILE()
|
||||
Count number of members answering a queue.
|
||||
</synopsis>
|
||||
<syntax>
|
||||
<parameter name="queuename" required="true" />
|
||||
<parameter name="queuename" required="false" />
|
||||
<parameter name="option" required="true">
|
||||
<enumlist>
|
||||
<enum name="logged">
|
||||
@@ -565,13 +565,22 @@ ASTERISK_REGISTER_FILE()
|
||||
<para>Returns the total number of members for the specified queue.</para>
|
||||
</enum>
|
||||
<enum name="penalty">
|
||||
<para>Gets or sets queue member penalty.</para>
|
||||
<para>Gets or sets queue member penalty. If
|
||||
<replaceable>queuename</replaceable> is not specified
|
||||
when setting the penalty then the penalty is set in all queues
|
||||
the interface is a member.</para>
|
||||
</enum>
|
||||
<enum name="paused">
|
||||
<para>Gets or sets queue member paused status.</para>
|
||||
<para>Gets or sets queue member paused status. If
|
||||
<replaceable>queuename</replaceable> is not specified
|
||||
when setting the paused status then the paused status is set
|
||||
in all queues the interface is a member.</para>
|
||||
</enum>
|
||||
<enum name="ringinuse">
|
||||
<para>Gets or sets queue member ringinuse.</para>
|
||||
<para>Gets or sets queue member ringinuse. If
|
||||
<replaceable>queuename</replaceable> is not specified
|
||||
when setting ringinuse then ringinuse is set
|
||||
in all queues the interface is a member.</para>
|
||||
</enum>
|
||||
</enumlist>
|
||||
</parameter>
|
||||
@@ -579,10 +588,8 @@ ASTERISK_REGISTER_FILE()
|
||||
</syntax>
|
||||
<description>
|
||||
<para>Allows access to queue counts [R] and member information [R/W].</para>
|
||||
<para>
|
||||
<replaceable>queuename</replaceable> is required for all operations
|
||||
<replaceable>interface</replaceable> is required for all member operations.
|
||||
</para>
|
||||
<para><replaceable>queuename</replaceable> is required for all read operations.</para>
|
||||
<para><replaceable>interface</replaceable> is required for all member operations.</para>
|
||||
</description>
|
||||
<see-also>
|
||||
<ref type="application">Queue</ref>
|
||||
@@ -8147,9 +8154,6 @@ static int queue_function_mem_read(struct ast_channel *chan, const char *cmd, ch
|
||||
static int queue_function_mem_write(struct ast_channel *chan, const char *cmd, char *data, const char *value)
|
||||
{
|
||||
int memvalue;
|
||||
struct call_queue *q;
|
||||
struct member *m;
|
||||
char rtvalue[80];
|
||||
|
||||
AST_DECLARE_APP_ARGS(args,
|
||||
AST_APP_ARG(queuename);
|
||||
@@ -8158,65 +8162,48 @@ static int queue_function_mem_write(struct ast_channel *chan, const char *cmd, c
|
||||
);
|
||||
|
||||
if (ast_strlen_zero(data)) {
|
||||
ast_log(LOG_ERROR, "Missing argument. QUEUE_MEMBER(<queuename>,<option>,<interface>)\n");
|
||||
ast_log(LOG_ERROR,
|
||||
"Missing required argument. %s([<queuename>],<option>,<interface>)\n",
|
||||
cmd);
|
||||
return -1;
|
||||
}
|
||||
|
||||
AST_STANDARD_APP_ARGS(args, data);
|
||||
|
||||
if (args.argc < 3) {
|
||||
ast_log(LOG_ERROR, "Missing argument. QUEUE_MEMBER_PENALTY(<queuename>,<interface>)\n");
|
||||
if (ast_strlen_zero(args.option)
|
||||
|| ast_strlen_zero(args.interface)) {
|
||||
ast_log(LOG_ERROR,
|
||||
"Missing required argument. %s([<queuename>],<option>,<interface>)\n",
|
||||
cmd);
|
||||
return -1;
|
||||
}
|
||||
|
||||
if (ast_strlen_zero(args.interface) && ast_strlen_zero(args.option)) {
|
||||
ast_log (LOG_ERROR, "<interface> and <option> parameter's can't be null\n");
|
||||
return -1;
|
||||
}
|
||||
/*
|
||||
* If queuename is empty then the option will be
|
||||
* set for the interface in all queues.
|
||||
*/
|
||||
|
||||
memvalue = atoi(value);
|
||||
if (!strcasecmp(args.option, "penalty")) {
|
||||
/* if queuename = NULL then penalty will be set for interface in all the queues.*/
|
||||
if (set_member_value(args.queuename, args.interface, MEMBER_PENALTY, memvalue)) {
|
||||
ast_log(LOG_ERROR, "Invalid interface, queue or penalty\n");
|
||||
ast_log(LOG_ERROR, "Invalid interface, queue, or penalty\n");
|
||||
return -1;
|
||||
}
|
||||
} else if ((q = find_load_queue_rt_friendly(args.queuename))) {
|
||||
ao2_lock(q);
|
||||
if ((m = interface_exists(q, args.interface))) {
|
||||
sprintf(rtvalue, "%s",(memvalue <= 0) ? "0" : "1");
|
||||
if (!strcasecmp(args.option, "paused")) {
|
||||
if (m->realtime) {
|
||||
update_realtime_member_field(m, q->name, args.option, rtvalue);
|
||||
}
|
||||
m->paused = (memvalue <= 0) ? 0 : 1;
|
||||
ast_devstate_changed(m->paused ? QUEUE_PAUSED_DEVSTATE : QUEUE_UNPAUSED_DEVSTATE,
|
||||
AST_DEVSTATE_CACHABLE, "Queue:%s_pause_%s", q->name, args.interface);
|
||||
|
||||
} else if ((!strcasecmp(args.option, "ignorebusy")) || (!strcasecmp(args.option, "ringinuse"))) {
|
||||
if (m->realtime) {
|
||||
update_realtime_member_field(m, q->name, args.option, rtvalue);
|
||||
}
|
||||
|
||||
m->ringinuse = (memvalue <= 0) ? 0 : 1;
|
||||
} else {
|
||||
ast_log(LOG_ERROR, "Invalid option, only penalty , paused or ringinuse/ignorebusy are valid\n");
|
||||
ao2_ref(m, -1);
|
||||
ao2_unlock(q);
|
||||
ao2_ref(q, -1);
|
||||
return -1;
|
||||
}
|
||||
ao2_ref(m, -1);
|
||||
} else {
|
||||
ao2_unlock(q);
|
||||
ao2_ref(q, -1);
|
||||
ast_log(LOG_ERROR, "Invalid interface for queue\n");
|
||||
} else if (!strcasecmp(args.option, "paused")) {
|
||||
memvalue = (memvalue <= 0) ? 0 : 1;
|
||||
if (set_member_paused(args.queuename, args.interface, NULL, memvalue)) {
|
||||
ast_log(LOG_ERROR, "Invalid interface or queue\n");
|
||||
return -1;
|
||||
}
|
||||
ao2_unlock(q);
|
||||
ao2_ref(q, -1);
|
||||
} else {
|
||||
ast_log(LOG_ERROR, "Invalid queue\n");
|
||||
} else if (!strcasecmp(args.option, "ignorebusy") /* ignorebusy is legacy */
|
||||
|| !strcasecmp(args.option, "ringinuse")) {
|
||||
memvalue = (memvalue <= 0) ? 0 : 1;
|
||||
if (set_member_value(args.queuename, args.interface, MEMBER_RINGINUSE, memvalue)) {
|
||||
ast_log(LOG_ERROR, "Invalid interface or queue\n");
|
||||
return -1;
|
||||
}
|
||||
} else {
|
||||
ast_log(LOG_ERROR, "%s: Invalid option '%s' provided.\n", cmd, args.option);
|
||||
return -1;
|
||||
}
|
||||
return 0;
|
||||
|
Reference in New Issue
Block a user