patch 9.1.0313: Crash when using heredoc with comment in command block

Problem:  Crash when using heredoc with comment in command block.
Solution: Handle a newline more like the end of the line, fix coverity
          warning (zeertzjq).

closes: #14535

Signed-off-by: zeertzjq <zeertzjq@outlook.com>
Signed-off-by: Yegappan Lakshmanan <yegappan@yahoo.com>
Signed-off-by: Christian Brabandt <cb@256bit.org>
This commit is contained in:
zeertzjq
2024-04-13 17:52:26 +02:00
committed by Christian Brabandt
parent e74cad3321
commit 1f5175d9af
6 changed files with 52 additions and 26 deletions

View File

@ -2088,17 +2088,6 @@ skiptowhite(char_u *p)
return p; return p;
} }
/*
* skiptowhite: skip over text until ' ' or '\t' or newline or NUL.
*/
char_u *
skiptowhite_or_nl(char_u *p)
{
while (*p != ' ' && *p != '\t' && *p != NL && *p != NUL)
++p;
return p;
}
/* /*
* skiptowhite_esc: Like skiptowhite(), but also skip escaped chars * skiptowhite_esc: Like skiptowhite(), but also skip escaped chars
*/ */

View File

@ -781,8 +781,15 @@ heredoc_get(exarg_T *eap, char_u *cmd, int script_get, int vim9compile)
int count = 0; int count = 0;
int heredoc_in_string = FALSE; int heredoc_in_string = FALSE;
char_u *line_arg = NULL; char_u *line_arg = NULL;
char_u *nl_ptr = vim_strchr(cmd, '\n');
if (eap->ea_getline == NULL && vim_strchr(cmd, '\n') == NULL) if (nl_ptr != NULL)
{
heredoc_in_string = TRUE;
line_arg = nl_ptr + 1;
*nl_ptr = NUL;
}
else if (eap->ea_getline == NULL)
{ {
emsg(_(e_cannot_use_heredoc_here)); emsg(_(e_cannot_use_heredoc_here));
return NULL; return NULL;
@ -826,14 +833,8 @@ heredoc_get(exarg_T *eap, char_u *cmd, int script_get, int vim9compile)
if (*cmd != NUL && *cmd != comment_char) if (*cmd != NUL && *cmd != comment_char)
{ {
marker = skipwhite(cmd); marker = skipwhite(cmd);
p = skiptowhite_or_nl(marker); p = skiptowhite(marker);
if (*p == NL) if (*skipwhite(p) != NUL && *skipwhite(p) != comment_char)
{
// heredoc in a string
line_arg = p + 1;
heredoc_in_string = TRUE;
}
else if (*skipwhite(p) != NUL && *skipwhite(p) != comment_char)
{ {
semsg(_(e_trailing_characters_str), p); semsg(_(e_trailing_characters_str), p);
return NULL; return NULL;

View File

@ -61,7 +61,6 @@ int vim_isalpha(int c);
int vim_toupper(int c); int vim_toupper(int c);
int vim_tolower(int c); int vim_tolower(int c);
char_u *skiptowhite(char_u *p); char_u *skiptowhite(char_u *p);
char_u *skiptowhite_or_nl(char_u *p);
char_u *skiptowhite_esc(char_u *p); char_u *skiptowhite_esc(char_u *p);
long getdigits(char_u **pp); long getdigits(char_u **pp);
long getdigits_quoted(char_u **pp); long getdigits_quoted(char_u **pp);

View File

@ -536,6 +536,13 @@ END
XX XX
call assert_equal(['Line1'], var1) call assert_equal(['Line1'], var1)
let var1 =<< trim XX " comment
Line1
Line2
Line3
XX
call assert_equal(['Line1', ' Line2', 'Line3'], var1)
" ignore "endfunc" " ignore "endfunc"
let var1 =<< END let var1 =<< END
something something
@ -716,15 +723,27 @@ END
call v9.CheckScriptFailure(lines, 'E15:') call v9.CheckScriptFailure(lines, 'E15:')
" Test for using heredoc in a single string using execute() " Test for using heredoc in a single string using execute()
call assert_equal(["['one', 'two']"], call assert_equal("\n['one', 'two']",
\ execute("let x =<< trim END\n one\n two\nEND\necho x")->split("\n")) \ execute("let x =<< trim END\n one\n two\nEND\necho x"))
call assert_equal(["[' one', ' two']"], call assert_equal("\n['one', ' two']",
\ execute("let x =<< END\n one\n two\nEND\necho x")->split("\n")) \ execute("let x =<< trim END\n one\n two\nEND\necho x"))
call assert_equal("\n['one', 'two']",
\ execute(" let x =<< trim END\n one\n two\n END\necho x"))
call assert_equal("\n['one', ' two']",
\ execute(" let x =<< trim END\n one\n two\n END\necho x"))
call assert_equal("\n[' one', ' two']",
\ execute("let x =<< END\n one\n two\nEND\necho x"))
call assert_equal("\n['one', 'two']",
\ execute("let x =<< END\none\ntwo\nEND\necho x"))
call assert_equal("\n['one', 'two']",
\ execute("let x =<< END \" comment\none\ntwo\nEND\necho x"))
let cmd = 'execute("let x =<< END\n one\n two\necho x")' let cmd = 'execute("let x =<< END\n one\n two\necho x")'
call assert_fails(cmd, "E990: Missing end marker 'END'") call assert_fails(cmd, "E990: Missing end marker 'END'")
let cmd = 'execute("let x =<<\n one\n two\necho x")' let cmd = 'execute("let x =<<\n one\n two\necho x")'
call assert_fails(cmd, "E990: Missing end marker ''") call assert_fails(cmd, "E172: Missing marker")
let cmd = 'execute("let x =<< trim\n one\n two\necho x")' let cmd = 'execute("let x =<< trim\n one\n two\necho x")'
call assert_fails(cmd, "E172: Missing marker")
let cmd = 'execute("let x =<< end\n one\n two\nend\necho x")'
call assert_fails(cmd, "E221: Marker cannot start with lower case letter") call assert_fails(cmd, "E221: Marker cannot start with lower case letter")
let cmd = 'execute("let x =<< eval END\n one\n two{y}\nEND\necho x")' let cmd = 'execute("let x =<< eval END\n one\n two{y}\nEND\necho x")'
call assert_fails(cmd, 'E121: Undefined variable: y') call assert_fails(cmd, 'E121: Undefined variable: y')

View File

@ -508,6 +508,22 @@ def Test_command_block_heredoc()
CODE CODE
v9.CheckSourceSuccess(lines) v9.CheckSourceSuccess(lines)
# Heredoc with comment
lines =<< trim CODE
vim9script
com SomeCommand {
g:someVar =<< trim END # comment
aaa
bbb
END
}
execute('SomeCommand')
assert_equal(['aaa', 'bbb'], g:someVar)
delcommand SomeCommand
unlet g:someVar
CODE
v9.CheckSourceSuccess(lines)
# heredoc evaluation # heredoc evaluation
lines =<< trim CODE lines =<< trim CODE
vim9script vim9script

View File

@ -704,6 +704,8 @@ static char *(features[]) =
static int included_patches[] = static int included_patches[] =
{ /* Add new patch number below this line */ { /* Add new patch number below this line */
/**/
313,
/**/ /**/
312, 312,
/**/ /**/