Commit d6edc71a69 for asterisk.org

commit d6edc71a694575151d15d11e10e75aa4c0c1e3c8
Author: serfreeman1337 <serfreeman1337@gmail.com>
Date:   Thu Jan 29 21:52:10 2026 +0500

    app_queue: Add 'prio' setting to the 'force_longest_waiting_caller' option

    This adds a 'prio' setting to ensure that call priority is respected across multiple queues.
    Using 'yes' could cause high-priority callers to be skipped if a caller
    in another queue had a longer wait time, regardless of priority.

    Resolves: #1637

    UserNote: The 'force_longest_waiting_caller' option now supports a 'prio' setting.
    When set to 'prio', calls are offered by priority first, then by wait time.

diff --git a/apps/app_queue.c b/apps/app_queue.c
index fdb426aa1b..b8ea7a616e 100644
--- a/apps/app_queue.c
+++ b/apps/app_queue.c
@@ -1942,6 +1942,10 @@ struct penalty_rule {
 #define ANNOUNCEPOSITION_MORE_THAN 3 /*!< We say "Currently there are more than <limit>" */
 #define ANNOUNCEPOSITION_LIMIT 4 /*!< We not announce position more than \<limit\> */

+#define FORCELONGESTWAITINGCALLER_NO 0
+#define FORCELONGESTWAITINGCALLER_YES 1
+#define FORCELONGESTWAITINGCALLER_PRIO 2 /*!< Account for call priorities when forcing longest waiting caller */
+
 struct call_queue {
 	AST_DECLARE_STRING_FIELDS(
 		/*! Queue name */
@@ -4778,7 +4782,13 @@ static int is_longest_waiting_caller(struct queue_ent *caller, struct member *me
 					 * will be unused until the first caller is picked up.
 					 */
 					if (!ch->pending) {
-						if (ch->start < caller->start) {
+						if (ch->prio != caller->prio && force_longest_waiting_caller == FORCELONGESTWAITINGCALLER_PRIO) {
+							if (ch->prio > caller->prio) { /* This queue has a caller with higher priority. */
+								ast_debug(1, "Queue %s has a call at position %i that's higher priority (%d vs %d)\n",
+									q->name, ch->pos, ch->prio, caller->prio);
+								is_longest_waiting = 0;
+							}
+						} else if (ch->start < caller->start) {
 							ast_debug(1, "Queue %s has a call at position %i that's been waiting longer (%li vs %li)\n",
 									  q->name, ch->pos, ch->start, caller->start);
 							is_longest_waiting = 0;
@@ -4787,6 +4797,8 @@ static int is_longest_waiting_caller(struct queue_ent *caller, struct member *me
 					}
 					ch = ch->next;
 				}
+
+				ao2_ref(mem, -1);
 			}
 		}
 		ao2_unlock(q);
@@ -9842,7 +9854,13 @@ static void queue_set_global_params(struct ast_config *cfg)
 		log_membername_as_agent = ast_true(general_val);
 	}
 	if ((general_val = ast_variable_retrieve(cfg, "general", "force_longest_waiting_caller"))) {
-		force_longest_waiting_caller = ast_true(general_val);
+		if (!strcasecmp(general_val, "prio")) {
+			force_longest_waiting_caller = FORCELONGESTWAITINGCALLER_PRIO;
+		} else if (ast_true(general_val)) {
+			force_longest_waiting_caller = FORCELONGESTWAITINGCALLER_YES;
+		} else {
+			force_longest_waiting_caller = FORCELONGESTWAITINGCALLER_NO;
+		}
 	}
 	if ((general_val = ast_variable_retrieve(cfg, "general", "log_unpause_on_reason_change"))) {
 		log_unpause_on_reason_change = ast_true(general_val);
diff --git a/configs/samples/queues.conf.sample b/configs/samples/queues.conf.sample
index 2b4cb4b097..16882e63c5 100644
--- a/configs/samples/queues.conf.sample
+++ b/configs/samples/queues.conf.sample
@@ -73,6 +73,8 @@ monitor-type = MixMonitor
 ; is a member of another queue with a call that's been waiting longer. If so, the current
 ; call is not offered to the agent. The default value is 'no'.
 ;
+; Setting this to 'prio' performs the same check but also accounts for call priority.
+;
 ;force_longest_waiting_caller = no
 ;
 ; Add unpause event to queue log in case of pause send twice with different reason code.