<p>Corey Farrell has uploaded this change for <strong>review</strong>.</p><p><a href="https://gerrit.asterisk.org/7560">View Change</a></p><pre style="font-family: monospace,monospace; white-space: pre-wrap;">CLI: Fix composite completion generators.<br><br>In cases where a parameter can be "all" or a list of channels we assume<br>that "all" will always match when generating the list of channels.<br><br>This issue is resolved by adding support for ast_cli_complete and<br>ast_complete_channels to use ast_cli_completion_add instead of returning<br>values.  Both functions switch modes when they are given -1 instead of<br>a->n.  Now adding "all" and adding the list of channels are two<br>independent operations.<br><br>ASTERISK-21038<br><br>Change-Id: Iebef61424689569338d0916ff2e37df944ac76fa<br>---<br>M channels/chan_iax2.c<br>M channels/chan_sip.c<br>M channels/chan_skinny.c<br>M include/asterisk/cli.h<br>M main/bridge.c<br>M main/cli.c<br>6 files changed, 49 insertions(+), 42 deletions(-)<br><br></pre><pre style="font-family: monospace,monospace; white-space: pre-wrap;">git pull ssh://gerrit.asterisk.org:29418/asterisk refs/changes/60/7560/1</pre><pre style="font-family: monospace,monospace; white-space: pre-wrap;">diff --git a/channels/chan_iax2.c b/channels/chan_iax2.c<br>index e618ee8..5de4286 100644<br>--- a/channels/chan_iax2.c<br>+++ b/channels/chan_iax2.c<br>@@ -3661,7 +3661,6 @@<br>      struct iax2_peer *peer = NULL;<br>        struct iax2_user *user = NULL;<br>        static const char * const choices[] = { "all", NULL };<br>-     char *cmplt;<br> <br>       switch (cmd) {<br>        case CLI_INIT:<br>@@ -3672,10 +3671,8 @@<br>                return NULL;<br>  case CLI_GENERATE:<br>            if (a->pos == 3) {<br>-                        cmplt = ast_cli_complete(a->word, choices, a->n);<br>-                      if (!cmplt)<br>-                          cmplt = complete_iax2_peers(a->line, a->word, a->pos, a->n - sizeof(choices), IAX_RTCACHEFRIENDS);<br>-                       return cmplt;<br>+                        ast_cli_complete(a->word, choices, -1);<br>+                   return complete_iax2_peers(a->line, a->word, a->pos, a->n, IAX_RTCACHEFRIENDS);<br>           }<br>             return NULL;<br>  }<br>diff --git a/channels/chan_sip.c b/channels/chan_sip.c<br>index 9add425..88c4148 100644<br>--- a/channels/chan_sip.c<br>+++ b/channels/chan_sip.c<br>@@ -20471,7 +20471,6 @@<br>       int havepattern = 0;<br>  struct ao2_iterator i;<br>        static const char * const choices[] = { "all", "like", NULL };<br>-   char *cmplt;<br> <br>       if (cmd == CLI_INIT) {<br>                e->command = "sip prune realtime [peer|all]";<br>@@ -20482,10 +20481,9 @@<br>          return NULL;<br>  } else if (cmd == CLI_GENERATE) {<br>             if (a->pos == 4 && !strcasecmp(a->argv[3], "peer")) {<br>-                        cmplt = ast_cli_complete(a->word, choices, a->n);<br>-                      if (!cmplt)<br>-                          cmplt = complete_sip_peer(a->word, a->n - sizeof(choices), SIP_PAGE2_RTCACHEFRIENDS);<br>-                  return cmplt;<br>+                        ast_cli_complete(a->word, choices, -1);<br>+<br>+                        return complete_sip_peer(a->word, a->n, SIP_PAGE2_RTCACHEFRIENDS);<br>              }<br>             if (a->pos == 5 && !strcasecmp(a->argv[4], "like"))<br>                   return complete_sip_peer(a->word, a->n, SIP_PAGE2_RTCACHEFRIENDS);<br>diff --git a/channels/chan_skinny.c b/channels/chan_skinny.c<br>index ccf6a94..922d4ce 100644<br>--- a/channels/chan_skinny.c<br>+++ b/channels/chan_skinny.c<br>@@ -3926,11 +3926,8 @@<br> {<br>         if (pos == 2) {<br>               static const char * const completions[] = { "all", NULL };<br>-         char *ret = ast_cli_complete(word, completions, state);<br>-              if (!ret) {<br>-                  ret = complete_skinny_devices(word, state - 1);<br>-              }<br>-            return ret;<br>+          ast_cli_complete(word, completions, -1);<br>+             return complete_skinny_devices(word, state);<br>  } else if (pos == 3) {<br>                static const char * const completions[] = { "restart", NULL };<br>              return ast_cli_complete(word, completions, state);<br>diff --git a/include/asterisk/cli.h b/include/asterisk/cli.h<br>index c2401a8..b4f60ac 100644<br>--- a/include/asterisk/cli.h<br>+++ b/include/asterisk/cli.h<br>@@ -201,6 +201,10 @@<br>  * Helper function to generate cli entries from a NULL-terminated array.<br>  * Returns the n-th matching entry from the array, or NULL if not found.<br>  * Can be used to implement generate() for static entries as below<br>+ *<br>+ * If 'pos' is less than 0 this function will always return NULL after all<br>+ * matching channels are added with ast_cli_completion_add.<br>+ *<br>  * (in this example we complete the word in position 2):<br>   \code<br>     char *my_generate(const char *line, const char *word, int pos, int n)<br>@@ -354,6 +358,9 @@<br>  * complete from the list of active channels.  'rpos' is the required<br>  * position in the command.  This function will return NULL immediately if<br>  * 'rpos' is not the same as the current position, 'pos'.<br>+ *<br>+ * If 'state' is less than 0 this function will always return NULL after all<br>+ * matching channels are added with ast_cli_completion_add.<br>  */<br> char *ast_complete_channels(const char *line, const char *word, int pos, int state, int rpos);<br> <br>diff --git a/main/bridge.c b/main/bridge.c<br>index f689b29..1c47d06 100644<br>--- a/main/bridge.c<br>+++ b/main/bridge.c<br>@@ -5196,7 +5196,6 @@<br> static char *handle_bridge_kick_channel(struct ast_cli_entry *e, int cmd, struct ast_cli_args *a)<br> {<br>      static const char * const completions[] = { "all", NULL };<br>- char *complete;<br>       struct ast_bridge *bridge;<br> <br>         switch (cmd) {<br>@@ -5213,11 +5212,9 @@<br>                        return complete_bridge_live(a->word, a->n);<br>             }<br>             if (a->pos == 3) {<br>-                        complete = ast_cli_complete(a->word, completions, a->n);<br>-                       if (!complete) {<br>-                             complete = complete_bridge_participant(a->argv[2], a->line, a->word, a->pos, a->n - 1);<br>-                       }<br>-                    return complete;<br>+                     ast_cli_complete(a->word, completions, -1);<br>+<br>+                    return complete_bridge_participant(a->argv[2], a->line, a->word, a->pos, a->n);<br>                }<br>             return NULL;<br>  }<br>diff --git a/main/cli.c b/main/cli.c<br>index 8e0cc3b..6ab7336 100644<br>--- a/main/cli.c<br>+++ b/main/cli.c<br>@@ -913,9 +913,12 @@<br>              return NULL;<br> <br>       case CLI_GENERATE:<br>-           if (a->pos != e->args)<br>+         if (a->pos != e->args) {<br>                        return NULL;<br>-         return ast_cli_complete(a->word, completions, a->n);<br>+           }<br>+            ast_cli_complete(a->word, completions, -1);<br>+<br>+            return NULL;<br>  }<br> <br>  /* regular handler */<br>@@ -1088,7 +1091,6 @@<br> {<br>      struct ast_channel *c = NULL;<br>         static const char * const completions[] = { "all", NULL };<br>- char *complete;<br> <br>    switch (cmd) {<br>        case CLI_INIT:<br>@@ -1104,11 +1106,10 @@<br>               if (a->pos != e->args) {<br>                        return NULL;<br>          }<br>-            complete = ast_cli_complete(a->word, completions, a->n);<br>-               if (!complete) {<br>-                     complete = ast_complete_channels(a->line, a->word, a->pos, a->n - 1, e->args);<br>-                }<br>-            return complete;<br>+             ast_cli_complete(a->word, completions, -1);<br>+               ast_complete_channels(a->line, a->word, a->pos, -1, e->args);<br>+<br>+         return NULL;<br>  }<br> <br>  if (a->argc != 4) {<br>@@ -1410,14 +1411,13 @@<br>               return NULL;<br>  case CLI_GENERATE:<br>            if (a->pos == 4) {<br>-                        char *complete = ast_cli_complete(a->word, completions_all, a->n);<br>-                     if (!complete) {<br>-                             complete = ast_complete_channels(a->line, a->word, a->pos, a->n - 1, e->args);<br>-                        }<br>-                    return complete;<br>+                     ast_cli_complete(a->word, completions_all, -1);<br>+                   ast_complete_channels(a->line, a->word, a->pos, -1, e->args);<br>             } else if (a->pos == 5) {<br>-                 return ast_cli_complete(a->word, completions_off, a->n);<br>+                       ast_cli_complete(a->word, completions_off, -1);<br>            }<br>+<br>+         return NULL;<br>  }<br> <br>  if (cmd == (CLI_HANDLER + 1000)) {<br>@@ -1508,7 +1508,9 @@<br>                     "       Shows lots of information about the specified channel.\n";<br>          return NULL;<br>  case CLI_GENERATE:<br>-           return ast_complete_channels(a->line, a->word, a->pos, a->n, 3);<br>+         ast_complete_channels(a->line, a->word, a->pos, a->n, -1);<br>+<br>+            return NULL;<br>  }<br> <br>  if (a->argc != 4) {<br>@@ -1658,8 +1660,13 @@<br>        len = ast_strlen_zero(word) ? 0 : strlen(word);<br> <br>    for (i = 0; choices[i]; i++) {<br>-               if ((!len || !strncasecmp(word, choices[i], len)) && ++which > state)<br>-                     return ast_strdup(choices[i]);<br>+               if ((!len || !strncasecmp(word, choices[i], len))) {<br>+                 if (state < 0) {<br>+                          ast_cli_completion_add(ast_strdup(choices[i]));<br>+                      } else if (++which > state) {<br>+                             return ast_strdup(choices[i]);<br>+                       }<br>+            }<br>     }<br>     return NULL;<br> }<br>@@ -1684,10 +1691,14 @@<br>     for (; (msg = ao2_iterator_next(&iter)); ao2_ref(msg, -1)) {<br>              struct ast_channel_snapshot *snapshot = stasis_message_data(msg);<br> <br>-         if (!strncasecmp(word, snapshot->name, wordlen) && (++which > state)) {<br>-                        ret = ast_strdup(snapshot->name);<br>-                 ao2_ref(msg, -1);<br>-                    break;<br>+               if (!strncasecmp(word, snapshot->name, wordlen)) {<br>+                        if (state < 0) {<br>+                          ast_cli_completion_add(ast_strdup(snapshot->name));<br>+                       } else if (++which > state) {<br>+                             ret = ast_strdup(snapshot->name);<br>+                         ao2_ref(msg, -1);<br>+                            break;<br>+                       }<br>             }<br>     }<br>     ao2_iterator_destroy(&iter);<br></pre><p>To view, visit <a href="https://gerrit.asterisk.org/7560">change 7560</a>. To unsubscribe, visit <a href="https://gerrit.asterisk.org/settings">settings</a>.</p><div itemscope itemtype="http://schema.org/EmailMessage"><div itemscope itemprop="action" itemtype="http://schema.org/ViewAction"><link itemprop="url" href="https://gerrit.asterisk.org/7560"/><meta itemprop="name" content="View Change"/></div></div>

<div style="display:none"> Gerrit-Project: asterisk </div>
<div style="display:none"> Gerrit-Branch: 13 </div>
<div style="display:none"> Gerrit-MessageType: newchange </div>
<div style="display:none"> Gerrit-Change-Id: Iebef61424689569338d0916ff2e37df944ac76fa </div>
<div style="display:none"> Gerrit-Change-Number: 7560 </div>
<div style="display:none"> Gerrit-PatchSet: 1 </div>
<div style="display:none"> Gerrit-Owner: Corey Farrell <git@cfware.com> </div>