summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorJason Cox <dev@jasoncarloscox.com>2021-08-29 12:36:49 +0200
committerBram Moolenaar <Bram@vim.org>2021-08-29 12:36:49 +0200
commit6e82351130ddb8d13cf3748b47f07cae77886fc7 (patch)
tree9007fdf6d5ed2bb0b05294beb034fe7206da11e8
parent9dcd349ca85e1df1abd1cbf1c29c0f1574b70bf8 (diff)
patch 8.2.3385: escaping for fish shell does not work properlyv8.2.3385
Problem: Escaping for fish shell does not work properly. Solution: Insert a backslash before a backslash. (Jason Cox, closes #8810)
-rw-r--r--runtime/doc/eval.txt4
-rw-r--r--src/strings.c21
-rw-r--r--src/testdir/test_shell.vim19
-rw-r--r--src/version.c2
4 files changed, 38 insertions, 8 deletions
diff --git a/runtime/doc/eval.txt b/runtime/doc/eval.txt
index db74f9c10b..93ff4169f9 100644
--- a/runtime/doc/eval.txt
+++ b/runtime/doc/eval.txt
@@ -10111,6 +10111,10 @@ shellescape({string} [, {special}]) *shellescape()*
escaped. When 'shell' containing "csh" in the tail it's
escaped a second time.
+ The "\" character will be escaped when 'shell' contains "fish"
+ in the tail. That is because for fish "\" is used as an escape
+ character inside single quotes.
+
Example of use with a |:!| command: >
:exe '!dir ' . shellescape(expand('<cfile>'), 1)
< This results in a directory listing for the file under the
diff --git a/src/strings.c b/src/strings.c
index ef0282a69b..566f4dab43 100644
--- a/src/strings.c
+++ b/src/strings.c
@@ -125,6 +125,15 @@ csh_like_shell(void)
}
/*
+ * Return TRUE when 'shell' has "fish" in the tail.
+ */
+ int
+fish_like_shell(void)
+{
+ return (strstr((char *)gettail(p_sh), "fish") != NULL);
+}
+
+/*
* Escape "string" for use as a shell argument with system().
* This uses single quotes, except when we know we need to use double quotes
* (MS-DOS and MS-Windows not using PowerShell and without 'shellslash' set).
@@ -145,6 +154,7 @@ vim_strsave_shellescape(char_u *string, int do_special, int do_newline)
char_u *escaped_string;
int l;
int csh_like;
+ int fish_like;
char_u *shname;
int powershell;
# ifdef MSWIN
@@ -157,6 +167,10 @@ vim_strsave_shellescape(char_u *string, int do_special, int do_newline)
// Csh also needs to have "\n" escaped twice when do_special is set.
csh_like = csh_like_shell();
+ // Fish shell uses '\' as an escape character within single quotes, so '\'
+ // itself must be escaped to get a literal '\'.
+ fish_like = fish_like_shell();
+
// PowerShell uses it's own version for quoting single quotes
shname = gettail(p_sh);
powershell = strstr((char *)shname, "pwsh") != NULL;
@@ -197,6 +211,8 @@ vim_strsave_shellescape(char_u *string, int do_special, int do_newline)
++length; // insert backslash
p += l - 1;
}
+ if (*p == '\\' && fish_like)
+ ++length; // insert backslash
}
// Allocate memory for the result and fill it.
@@ -261,6 +277,11 @@ vim_strsave_shellescape(char_u *string, int do_special, int do_newline)
*d++ = *p++;
continue;
}
+ if (*p == '\\' && fish_like)
+ {
+ *d++ = '\\';
+ *d++ = *p++;
+ }
MB_COPY_CHAR(p, d);
}
diff --git a/src/testdir/test_shell.vim b/src/testdir/test_shell.vim
index bdc6631777..fa11f08d72 100644
--- a/src/testdir/test_shell.vim
+++ b/src/testdir/test_shell.vim
@@ -61,18 +61,21 @@ func Test_shell_options()
for e in shells
exe 'set shell=' .. e[0]
if e[0] =~# '.*csh$' || e[0] =~# '.*csh.exe$'
- let str1 = "'cmd \"arg1\" '\\''arg2'\\'' \\!%#'"
- let str2 = "'cmd \"arg1\" '\\''arg2'\\'' \\\\!\\%\\#'"
+ let str1 = "'cmd \"arg1\" '\\''arg2'\\'' \\!%# \\'"
+ let str2 = "'cmd \"arg1\" '\\''arg2'\\'' \\\\!\\%\\# \\'"
elseif e[0] =~# '.*powershell$' || e[0] =~# '.*powershell.exe$'
\ || e[0] =~# '.*pwsh$' || e[0] =~# '.*pwsh.exe$'
- let str1 = "'cmd \"arg1\" ''arg2'' !%#'"
- let str2 = "'cmd \"arg1\" ''arg2'' \\!\\%\\#'"
+ let str1 = "'cmd \"arg1\" ''arg2'' !%# \\'"
+ let str2 = "'cmd \"arg1\" ''arg2'' \\!\\%\\# \\'"
+ elseif e[0] =~# '.*fish$' || e[0] =~# '.*fish.exe$'
+ let str1 = "'cmd \"arg1\" '\\''arg2'\\'' !%# \\\\'"
+ let str2 = "'cmd \"arg1\" '\\''arg2'\\'' \\!\\%\\# \\\\'"
else
- let str1 = "'cmd \"arg1\" '\\''arg2'\\'' !%#'"
- let str2 = "'cmd \"arg1\" '\\''arg2'\\'' \\!\\%\\#'"
+ let str1 = "'cmd \"arg1\" '\\''arg2'\\'' !%# \\'"
+ let str2 = "'cmd \"arg1\" '\\''arg2'\\'' \\!\\%\\# \\'"
endif
- call assert_equal(str1, shellescape("cmd \"arg1\" 'arg2' !%#"), e[0])
- call assert_equal(str2, shellescape("cmd \"arg1\" 'arg2' !%#", 1), e[0])
+ call assert_equal(str1, shellescape("cmd \"arg1\" 'arg2' !%# \\"), e[0])
+ call assert_equal(str2, shellescape("cmd \"arg1\" 'arg2' !%# \\", 1), e[0])
" Try running an external command with the shell.
if executable(e[0])
diff --git a/src/version.c b/src/version.c
index d50d09829d..5f427a8774 100644
--- a/src/version.c
+++ b/src/version.c
@@ -756,6 +756,8 @@ static char *(features[]) =
static int included_patches[] =
{ /* Add new patch number below this line */
/**/
+ 3385,
+/**/
3384,
/**/
3383,