patch 8.2.3916: no error for passing an invalid line number to append()

Problem:    No error for passing an invalid line number to append().
Solution:   In Vim9 script check for a non-negative number. (closes #9417)
This commit is contained in:
Bram Moolenaar
2021-12-27 20:57:06 +00:00
parent 4b28ba3245
commit 8dac2acd6a
7 changed files with 31 additions and 8 deletions

View File

@ -357,6 +357,10 @@ EXTERN char e_using_invalid_value_as_string_str[]
EXTERN char e_cannot_index_special_variable[] EXTERN char e_cannot_index_special_variable[]
INIT(= N_("E909: Cannot index a special variable")); INIT(= N_("E909: Cannot index a special variable"));
#endif #endif
EXTERN char_u e_invalid_column_number_nr[]
INIT(= N_("E964: Invalid column number: %ld"));
EXTERN char_u e_invalid_line_number_nr[]
INIT(= N_("E966: Invalid line number: %ld"));
EXTERN char e_command_not_supported_in_vim9_script_missing_var_str[] EXTERN char e_command_not_supported_in_vim9_script_missing_var_str[]
INIT(= N_("E1100: Command not supported in Vim9 script (missing :var?): %s")); INIT(= N_("E1100: Command not supported in Vim9 script (missing :var?): %s"));
#ifdef FEAT_EVAL #ifdef FEAT_EVAL

View File

@ -5359,8 +5359,9 @@ var2fpos(
name = tv_get_string_chk(varp); name = tv_get_string_chk(varp);
if (name == NULL) if (name == NULL)
return NULL; return NULL;
if (name[0] == '.') // cursor if (name[0] == '.' && (!in_vim9script() || name[1] == NUL))
{ {
// cursor
pos = curwin->w_cursor; pos = curwin->w_cursor;
if (charcol) if (charcol)
pos.col = buf_byteidx_to_charidx(curbuf, pos.lnum, pos.col); pos.col = buf_byteidx_to_charidx(curbuf, pos.lnum, pos.col);
@ -5376,8 +5377,10 @@ var2fpos(
pos.col = buf_byteidx_to_charidx(curbuf, pos.lnum, pos.col); pos.col = buf_byteidx_to_charidx(curbuf, pos.lnum, pos.col);
return &pos; return &pos;
} }
if (name[0] == '\'') // mark if (name[0] == '\'' && (!in_vim9script()
|| (name[1] != NUL && name[2] == NUL)))
{ {
// mark
pp = getmark_buf_fnum(curbuf, name[1], FALSE, fnum); pp = getmark_buf_fnum(curbuf, name[1], FALSE, fnum);
if (pp == NULL || pp == (pos_T *)-1 || pp->lnum <= 0) if (pp == NULL || pp == (pos_T *)-1 || pp->lnum <= 0)
return NULL; return NULL;

View File

@ -151,6 +151,8 @@ set_buffer_lines(
if (buf == NULL || (!is_curbuf && buf->b_ml.ml_mfp == NULL) || lnum < 1) if (buf == NULL || (!is_curbuf && buf->b_ml.ml_mfp == NULL) || lnum < 1)
{ {
rettv->vval.v_number = 1; // FAIL rettv->vval.v_number = 1; // FAIL
if (in_vim9script() && lnum < 1)
semsg(_(e_invalid_line_number_nr), lnum_arg);
return; return;
} }

View File

@ -2130,8 +2130,12 @@ f_indent(typval_T *argvars, typval_T *rettv)
if (lnum >= 1 && lnum <= curbuf->b_ml.ml_line_count) if (lnum >= 1 && lnum <= curbuf->b_ml.ml_line_count)
rettv->vval.v_number = get_indent_lnum(lnum); rettv->vval.v_number = get_indent_lnum(lnum);
else else
{
if (in_vim9script())
semsg(_(e_invalid_line_number_nr), lnum);
rettv->vval.v_number = -1; rettv->vval.v_number = -1;
} }
}
/* /*
* "lispindent(lnum)" function * "lispindent(lnum)" function
@ -2154,6 +2158,8 @@ f_lispindent(typval_T *argvars UNUSED, typval_T *rettv)
rettv->vval.v_number = get_lisp_indent(); rettv->vval.v_number = get_lisp_indent();
curwin->w_cursor = pos; curwin->w_cursor = pos;
} }
else if (in_vim9script())
semsg(_(e_invalid_line_number_nr), lnum);
else else
#endif #endif
rettv->vval.v_number = -1; rettv->vval.v_number = -1;

View File

@ -182,6 +182,9 @@ def Test_append()
assert_equal("function('min')", getline(1)) assert_equal("function('min')", getline(1))
CheckDefAndScriptFailure(['append([1], "x")'], ['E1013: Argument 1: type mismatch, expected string but got list<number>', 'E1220: String or Number required for argument 1']) CheckDefAndScriptFailure(['append([1], "x")'], ['E1013: Argument 1: type mismatch, expected string but got list<number>', 'E1220: String or Number required for argument 1'])
CheckDefExecAndScriptFailure(['append("", "x")'], 'E1209: Invalid value for a line number') CheckDefExecAndScriptFailure(['append("", "x")'], 'E1209: Invalid value for a line number')
CheckDefExecAndScriptFailure(['append(".a", "x")'], 'E1209: Invalid value for a line number')
CheckDefExecAndScriptFailure(['append("''aa", "x")'], 'E1209: Invalid value for a line number')
CheckDefExecAndScriptFailure(['append(-1, "x")'], 'E966: Invalid line number: -1')
bwipe! bwipe!
enddef enddef
@ -199,6 +202,7 @@ def Test_appendbufline()
assert_equal(['zero'], getbufline(bnum, 1)) assert_equal(['zero'], getbufline(bnum, 1))
CheckDefAndScriptFailure(['appendbufline([1], 1, "x")'], ['E1013: Argument 1: type mismatch, expected string but got list<number>', 'E1220: String or Number required for argument 1']) CheckDefAndScriptFailure(['appendbufline([1], 1, "x")'], ['E1013: Argument 1: type mismatch, expected string but got list<number>', 'E1220: String or Number required for argument 1'])
CheckDefAndScriptFailure(['appendbufline(1, [1], "x")'], ['E1013: Argument 2: type mismatch, expected string but got list<number>', 'E1220: String or Number required for argument 2']) CheckDefAndScriptFailure(['appendbufline(1, [1], "x")'], ['E1013: Argument 2: type mismatch, expected string but got list<number>', 'E1220: String or Number required for argument 2'])
CheckDefExecAndScriptFailure(['appendbufline(' .. bnum .. ', -1, "x")'], 'E966: Invalid line number: -1')
CheckDefAndScriptFailure(['appendbufline(1, 1, {"a": 10})'], ['E1013: Argument 3: type mismatch, expected string but got dict<number>', 'E1224: String, Number or List required for argument 3']) CheckDefAndScriptFailure(['appendbufline(1, 1, {"a": 10})'], ['E1013: Argument 3: type mismatch, expected string but got dict<number>', 'E1224: String, Number or List required for argument 3'])
bnum->bufwinid()->win_gotoid() bnum->bufwinid()->win_gotoid()
appendbufline('', 0, 'numbers') appendbufline('', 0, 'numbers')
@ -1834,6 +1838,7 @@ def Test_indent()
CheckDefAndScriptFailure(['indent([1])'], ['E1013: Argument 1: type mismatch, expected string but got list<number>', 'E1220: String or Number required for argument 1']) CheckDefAndScriptFailure(['indent([1])'], ['E1013: Argument 1: type mismatch, expected string but got list<number>', 'E1220: String or Number required for argument 1'])
CheckDefAndScriptFailure(['indent(true)'], ['E1013: Argument 1: type mismatch, expected string but got bool', 'E1220: String or Number required for argument 1']) CheckDefAndScriptFailure(['indent(true)'], ['E1013: Argument 1: type mismatch, expected string but got bool', 'E1220: String or Number required for argument 1'])
CheckDefExecAndScriptFailure(['indent("")'], 'E1209: Invalid value for a line number') CheckDefExecAndScriptFailure(['indent("")'], 'E1209: Invalid value for a line number')
CheckDefExecAndScriptFailure(['indent(-1)'], 'E966: Invalid line number: -1')
assert_equal(0, indent(1)) assert_equal(0, indent(1))
enddef enddef
@ -2061,6 +2066,7 @@ enddef
def Test_lispindent() def Test_lispindent()
CheckDefAndScriptFailure(['lispindent({})'], ['E1013: Argument 1: type mismatch, expected string but got dict<unknown>', 'E1220: String or Number required for argument 1']) CheckDefAndScriptFailure(['lispindent({})'], ['E1013: Argument 1: type mismatch, expected string but got dict<unknown>', 'E1220: String or Number required for argument 1'])
CheckDefExecAndScriptFailure(['lispindent("")'], 'E1209: Invalid value for a line number') CheckDefExecAndScriptFailure(['lispindent("")'], 'E1209: Invalid value for a line number')
CheckDefExecAndScriptFailure(['lispindent(-1)'], 'E966: Invalid line number: -1')
assert_equal(0, lispindent(1)) assert_equal(0, lispindent(1))
enddef enddef
@ -3239,6 +3245,7 @@ def Test_setbufline()
assert_equal(['1', '2', '3', 'one', '10', 'two', '11'], getbufline(bnum, 1, '$')) assert_equal(['1', '2', '3', 'one', '10', 'two', '11'], getbufline(bnum, 1, '$'))
CheckDefAndScriptFailure(['setbufline([1], 1, "x")'], ['E1013: Argument 1: type mismatch, expected string but got list<number>', 'E1220: String or Number required for argument 1']) CheckDefAndScriptFailure(['setbufline([1], 1, "x")'], ['E1013: Argument 1: type mismatch, expected string but got list<number>', 'E1220: String or Number required for argument 1'])
CheckDefAndScriptFailure(['setbufline(1, [1], "x")'], ['E1013: Argument 2: type mismatch, expected string but got list<number>', 'E1220: String or Number required for argument 2']) CheckDefAndScriptFailure(['setbufline(1, [1], "x")'], ['E1013: Argument 2: type mismatch, expected string but got list<number>', 'E1220: String or Number required for argument 2'])
CheckDefExecAndScriptFailure(['setbufline(' .. bnum .. ', -1, "x")'], 'E966: Invalid line number: -1')
CheckDefAndScriptFailure(['setbufline(1, 1, {"a": 10})'], ['E1013: Argument 3: type mismatch, expected string but got dict<number>', 'E1224: String, Number or List required for argument 3']) CheckDefAndScriptFailure(['setbufline(1, 1, {"a": 10})'], ['E1013: Argument 3: type mismatch, expected string but got dict<number>', 'E1224: String, Number or List required for argument 3'])
bnum->bufwinid()->win_gotoid() bnum->bufwinid()->win_gotoid()
setbufline('', 1, 'nombres') setbufline('', 1, 'nombres')
@ -3303,6 +3310,7 @@ def Test_setline()
assert_equal(['10', 'b', 'c', 'd'], getline(1, '$')) assert_equal(['10', 'b', 'c', 'd'], getline(1, '$'))
CheckDefAndScriptFailure(['setline([1], "x")'], ['E1013: Argument 1: type mismatch, expected string but got list<number>', 'E1220: String or Number required for argument 1']) CheckDefAndScriptFailure(['setline([1], "x")'], ['E1013: Argument 1: type mismatch, expected string but got list<number>', 'E1220: String or Number required for argument 1'])
CheckDefExecAndScriptFailure(['setline("", "x")'], 'E1209: Invalid value for a line number') CheckDefExecAndScriptFailure(['setline("", "x")'], 'E1209: Invalid value for a line number')
CheckDefExecAndScriptFailure(['setline(-1, "x")'], 'E966: Invalid line number: -1')
bw! bw!
enddef enddef

View File

@ -48,8 +48,6 @@ static hashtab_T *global_proptypes = NULL;
static int proptype_id = 0; static int proptype_id = 0;
static char_u e_type_not_exist[] = N_("E971: Property type %s does not exist"); static char_u e_type_not_exist[] = N_("E971: Property type %s does not exist");
static char_u e_invalid_col[] = N_("E964: Invalid column number: %ld");
static char_u e_invalid_lnum[] = N_("E966: Invalid line number: %ld");
/* /*
* Find a property type by name, return the hashitem. * Find a property type by name, return the hashitem.
@ -169,7 +167,7 @@ f_prop_add(typval_T *argvars, typval_T *rettv UNUSED)
start_col = tv_get_number(&argvars[1]); start_col = tv_get_number(&argvars[1]);
if (start_col < 1) if (start_col < 1)
{ {
semsg(_(e_invalid_col), (long)start_col); semsg(_(e_invalid_column_number_nr), (long)start_col);
return; return;
} }
if (argvars[2].v_type != VAR_DICT) if (argvars[2].v_type != VAR_DICT)
@ -213,12 +211,12 @@ prop_add_one(
if (start_lnum < 1 || start_lnum > buf->b_ml.ml_line_count) if (start_lnum < 1 || start_lnum > buf->b_ml.ml_line_count)
{ {
semsg(_(e_invalid_lnum), (long)start_lnum); semsg(_(e_invalid_line_number_nr), (long)start_lnum);
return FAIL; return FAIL;
} }
if (end_lnum < start_lnum || end_lnum > buf->b_ml.ml_line_count) if (end_lnum < start_lnum || end_lnum > buf->b_ml.ml_line_count)
{ {
semsg(_(e_invalid_lnum), (long)end_lnum); semsg(_(e_invalid_line_number_nr), (long)end_lnum);
return FAIL; return FAIL;
} }
@ -243,7 +241,7 @@ prop_add_one(
col = 1; col = 1;
if (col - 1 > (colnr_T)textlen) if (col - 1 > (colnr_T)textlen)
{ {
semsg(_(e_invalid_col), (long)start_col); semsg(_(e_invalid_column_number_nr), (long)start_col);
return FAIL; return FAIL;
} }

View File

@ -749,6 +749,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 */
/**/
3916,
/**/ /**/
3915, 3915,
/**/ /**/