# HG changeset patch # User Christian Brabandt # Date 1527001206 -7200 # Node ID c3f9c37160e7893af653de3cd8e990cc91d761b4 # Parent 4400705e825ab3fb98336c661bc9d025157dafbb patch 8.1.0017: shell command completion has duplicates commit https://github.com/vim/vim/commit/62fe66f251263715968442e237742d9d3dfd5fa1 Author: Bram Moolenaar Date: Tue May 22 16:58:47 2018 +0200 patch 8.1.0017: shell command completion has duplicates Problem: Shell command completion has duplicates. (Yegappan Lakshmanan) Solution: Use a hash table to avoid duplicates. (Ozaki Kiichi, closes https://github.com/vim/vim/issues/539, closes #2733) diff --git a/src/ex_getln.c b/src/ex_getln.c --- a/src/ex_getln.c +++ b/src/ex_getln.c @@ -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; } diff --git a/src/testdir/test_cmdline.vim b/src/testdir/test_cmdline.vim --- a/src/testdir/test_cmdline.vim +++ b/src/testdir/test_cmdline.vim @@ -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 diff --git a/src/version.c b/src/version.c --- a/src/version.c +++ b/src/version.c @@ -762,6 +762,8 @@ static char *(features[]) = static int included_patches[] = { /* Add new patch number below this line */ /**/ + 17, +/**/ 16, /**/ 15,