summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorBram Moolenaar <Bram@vim.org>2018-05-22 16:58:47 +0200
committerBram Moolenaar <Bram@vim.org>2018-05-22 16:58:47 +0200
commit62fe66f251263715968442e237742d9d3dfd5fa1 (patch)
tree17a95bf476e85695ea332ef5bc2e50371d66622d
parentd45aa55d42211eb2f42e0a04ff77537a5df51c40 (diff)
patch 8.1.0017: shell command completion has duplicatesv8.1.0017
Problem: Shell command completion has duplicates. (Yegappan Lakshmanan) Solution: Use a hash table to avoid duplicates. (Ozaki Kiichi, closes #539, closes #2733)
-rw-r--r--src/ex_getln.c43
-rw-r--r--src/testdir/test_cmdline.vim12
-rw-r--r--src/version.c2
3 files changed, 38 insertions, 19 deletions
diff --git a/src/ex_getln.c b/src/ex_getln.c
index acaaef4705..c4b9acd75d 100644
--- 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
index 473c35dcf9..c807364ec3 100644
--- 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
index 7d941bbe44..4e4c1fbee3 100644
--- 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,