]> granicus.if.org Git - vim/commitdiff
patch 8.1.0017: shell command completion has duplicates v8.1.0017
authorBram Moolenaar <Bram@vim.org>
Tue, 22 May 2018 14:58:47 +0000 (16:58 +0200)
committerBram Moolenaar <Bram@vim.org>
Tue, 22 May 2018 14:58:47 +0000 (16:58 +0200)
Problem:    Shell command completion has duplicates. (Yegappan Lakshmanan)
Solution:   Use a hash table to avoid duplicates. (Ozaki Kiichi, closes #539,
            closes #2733)

src/ex_getln.c
src/testdir/test_cmdline.vim
src/version.c

index acaaef4705bf6f86781a65cd476d079c343df63e..c4b9acd75d1da5272973a11dd17aa5cab1d5b8b5 100644 (file)
@@ -5147,7 +5147,7 @@ expand_shellcmd(
 {
     char_u     *pat;
     int                i;
-    char_u     *path;
+    char_u     *path = NULL;
     int                mustfree = FALSE;
     garray_T    ga;
     char_u     *buf = alloc(MAXPATHL);
@@ -5156,6 +5156,9 @@ expand_shellcmd(
     int                flags = flagsarg;
     int                ret;
     int                did_curdir = FALSE;
+    hashtab_T  found_ht;
+    hashitem_T *hi;
+    hash_T     hash;
 
     if (buf == NULL)
        return FAIL;
@@ -5169,15 +5172,14 @@ expand_shellcmd(
 
     flags |= EW_FILE | EW_EXEC | EW_SHELLCMD;
 
-    /* For an absolute name we don't use $PATH. */
-    if (mch_isFullName(pat))
-       path = (char_u *)" ";
-    else if ((pat[0] == '.' && (vim_ispathsep(pat[1])
-                           || (pat[1] == '.' && vim_ispathsep(pat[2])))))
+    if (pat[0] == '.' && (vim_ispathsep(pat[1])
+                              || (pat[1] == '.' && vim_ispathsep(pat[2]))))
        path = (char_u *)".";
     else
     {
-       path = vim_getenv((char_u *)"PATH", &mustfree);
+       /* For an absolute name we don't use $PATH. */
+       if (!mch_isFullName(pat))
+           path = vim_getenv((char_u *)"PATH", &mustfree);
        if (path == NULL)
            path = (char_u *)"";
     }
@@ -5188,6 +5190,7 @@ expand_shellcmd(
      * current directory, to find "subdir/cmd".
      */
     ga_init2(&ga, (int)sizeof(char *), 10);
+    hash_init(&found_ht);
     for (s = path; ; s = e)
     {
        if (*s == NUL)
@@ -5200,9 +5203,6 @@ expand_shellcmd(
        else if (*s == '.')
            did_curdir = TRUE;
 
-       if (*s == ' ')
-           ++s;        /* Skip space used for absolute path name. */
-
 #if defined(MSWIN)
        e = vim_strchr(s, ';');
 #else
@@ -5229,15 +5229,23 @@ expand_shellcmd(
            {
                for (i = 0; i < *num_file; ++i)
                {
-                   s = (*file)[i];
-                   if (STRLEN(s) > l)
+                   char_u *name = (*file)[i];
+
+                   if (STRLEN(name) > l)
                    {
-                       /* Remove the path again. */
-                       STRMOVE(s, s + l);
-                       ((char_u **)ga.ga_data)[ga.ga_len++] = s;
+                       // Check if this name was already found.
+                       hash = hash_hash(name + l);
+                       hi = hash_lookup(&found_ht, name + l, hash);
+                       if (HASHITEM_EMPTY(hi))
+                       {
+                           // Remove the path that was prepended.
+                           STRMOVE(name, name + l);
+                           ((char_u **)ga.ga_data)[ga.ga_len++] = name;
+                           hash_add_item(&found_ht, hi, name, hash);
+                           name = NULL;
+                       }
                    }
-                   else
-                       vim_free(s);
+                   vim_free(name);
                }
                vim_free(*file);
            }
@@ -5252,6 +5260,7 @@ expand_shellcmd(
     vim_free(pat);
     if (mustfree)
        vim_free(path);
+    hash_clear(&found_ht);
     return OK;
 }
 
index 473c35dcf9cac57d3d8a0a371bda5a591b3ca74a..c807364ec335e9e2d8bf19737597326f4f9349f2 100644 (file)
@@ -231,6 +231,15 @@ func Test_getcompletion()
   let l = getcompletion('not', 'mapclear')
   call assert_equal([], l)
 
+  let l = getcompletion('.', 'shellcmd')
+  call assert_equal(['./', '../'], l[0:1])
+  call assert_equal(-1, match(l[2:], '^\.\.\?/$'))
+  let root = has('win32') ? 'C:\\' : '/'
+  let l = getcompletion(root, 'shellcmd')
+  let expected = map(filter(glob(root . '*', 0, 1),
+        \ 'isdirectory(v:val) || executable(v:val)'), 'isdirectory(v:val) ? v:val . ''/'' : v:val')
+  call assert_equal(expected, l)
+
   if has('cscope')
     let l = getcompletion('', 'cscope')
     let cmds = ['add', 'find', 'help', 'kill', 'reset', 'show']
@@ -258,8 +267,7 @@ func Test_getcompletion()
   endif
 
   " For others test if the name is recognized.
-  let names = ['buffer', 'environment', 'file_in_path',
-       \ 'mapping', 'shellcmd', 'tag', 'tag_listfiles', 'user']
+  let names = ['buffer', 'environment', 'file_in_path', 'mapping', 'tag', 'tag_listfiles', 'user']
   if has('cmdline_hist')
     call add(names, 'history')
   endif
index 7d941bbe4462e898c06745baea61b2edf4a376e3..4e4c1fbee38c1e7d064711ce6659fcb867b2e8f1 100644 (file)
@@ -761,6 +761,8 @@ static char *(features[]) =
 
 static int included_patches[] =
 {   /* Add new patch number below this line */
+/**/
+    17,
 /**/
     16,
 /**/