blob: e955064074dd7490967ff7f2dd2433f41144186c [file] [log] [blame]
" Test Vim9 assignments
import './util/vim9.vim' as v9
let s:appendToMe = 'xxx'
let s:addToMe = 111
let s:newVar = ''
let g:existing = 'yes'
let g:inc_counter = 1
let $SOME_ENV_VAR = 'some'
let g:alist = [7]
let g:adict = #{a: 1}
let g:astring = 'text'
def Test_assignment_bool()
var bool1: bool = true
assert_equal(v:true, bool1)
var bool2: bool = false
assert_equal(v:false, bool2)
var bool3: bool = 0
assert_equal(false, bool3)
var bool4: bool = 1
assert_equal(true, bool4)
var bool5: bool = 1 && true
assert_equal(true, bool5)
var bool6: bool = 0 && 1
assert_equal(false, bool6)
var bool7: bool = 0 || 1 && true
assert_equal(true, bool7)
var lines =<< trim END
vim9script
def GetFlag(): bool
var flag: bool = 1
return flag
enddef
var flag: bool = GetFlag()
assert_equal(true, flag)
flag = 0
assert_equal(false, flag)
flag = 1
assert_equal(true, flag)
flag = 1 || true
assert_equal(true, flag)
flag = 1 && false
assert_equal(false, flag)
var cp: bool = &cp
var fen: bool = &l:fen
END
v9.CheckScriptSuccess(lines)
v9.CheckDefAndScriptFailure(['var x: bool = 2'], 'E1012:')
v9.CheckDefAndScriptFailure(['var x: bool = -1'], 'E1012:')
v9.CheckDefAndScriptFailure(['var x: bool = [1]'], 'E1012:')
v9.CheckDefAndScriptFailure(['var x: bool = {}'], 'E1012:')
v9.CheckDefAndScriptFailure(['var x: bool = "x"'], 'E1012:')
v9.CheckDefAndScriptFailure(['var x: bool = "x"', '', 'eval 0'], 'E1012:', 1)
enddef
def Test_syntax()
var name = 234
var other: list<string> = ['asdf']
enddef
def Test_assignment()
v9.CheckDefFailure(['var x:string'], 'E1069:')
v9.CheckDefFailure(['var x:string = "x"'], 'E1069:')
v9.CheckDefFailure(['var a:string = "x"'], 'E1069:')
v9.CheckDefFailure(['var lambda = () => "lambda"'], 'E704:')
v9.CheckScriptFailure(['var x = "x"'], 'E1124:')
# lower case name is OK for a list
var lambdaLines =<< trim END
var lambdaList: list<func> = [g:Test_syntax]
lambdaList[0] = () => "lambda"
END
v9.CheckDefAndScriptSuccess(lambdaLines)
var nr: number = 1234
v9.CheckDefFailure(['var nr: number = "asdf"'], 'E1012:')
var a: number = 6 #comment
assert_equal(6, a)
if has('channel')
var chan1: channel
assert_equal('fail', ch_status(chan1))
var job1: job
assert_equal('fail', job_status(job1))
# calling job_start() is in test_vim9_fails.vim, it causes leak reports
endif
var float1: float = 3.4
var Funky1: func
var Funky2: func = function('len')
var Party2: func = funcref('g:Test_syntax')
g:newvar = 'new' #comment
assert_equal('new', g:newvar)
assert_equal('yes', g:existing)
g:existing = 'no'
assert_equal('no', g:existing)
v:char = 'abc'
assert_equal('abc', v:char)
$ENVVAR = 'foobar'
assert_equal('foobar', $ENVVAR)
$ENVVAR = ''
var lines =<< trim END
vim9script
$ENVVAR = 'barfoo'
assert_equal('barfoo', $ENVVAR)
$ENVVAR = ''
END
v9.CheckScriptSuccess(lines)
appendToMe ..= 'yyy'
assert_equal('xxxyyy', appendToMe)
addToMe += 222
assert_equal(333, addToMe)
newVar = 'new'
assert_equal('new', newVar)
set ts=7
var ts: number = &ts
assert_equal(7, ts)
&ts += 1
assert_equal(8, &ts)
&ts -= 3
assert_equal(5, &ts)
&ts *= 2
assert_equal(10, &ts)
&ts /= 3
assert_equal(3, &ts)
set ts=10
&ts %= 4
assert_equal(2, &ts)
assert_fails('&ts /= 0', ['E1154:', 'E1154:'])
assert_fails('&ts %= 0', ['E1154:', 'E1154:'])
assert_fails('&ts /= []', ['E745:', 'E745:'])
assert_fails('&ts %= []', ['E745:', 'E745:'])
assert_equal(2, &ts)
var f100: float = 100.0
f100 /= 5
assert_equal(20.0, f100)
var f200: float = 200.0
f200 /= 5.0
assert_equal(40.0, f200)
v9.CheckDefFailure(['var nr: number = 200', 'nr /= 5.0'], 'E1012:')
lines =<< trim END
&ts = 6
&ts += 3
assert_equal(9, &ts)
&l:ts = 6
assert_equal(6, &ts)
&l:ts += 2
assert_equal(8, &ts)
&g:ts = 6
assert_equal(6, &g:ts)
&g:ts += 2
assert_equal(8, &g:ts)
&number = true
assert_equal(true, &number)
&number = 0
assert_equal(false, &number)
&number = 1
assert_equal(true, &number)
&number = false
assert_equal(false, &number)
END
v9.CheckDefAndScriptSuccess(lines)
v9.CheckDefFailure(['&notex += 3'], 'E113:')
v9.CheckDefFailure(['&ts ..= "xxx"'], 'E1019:')
v9.CheckDefFailure(['var d = {k: [0]}', 'd.k ..= "x"'], 'E1012: Type mismatch; expected list<number> but got string')
v9.CheckDefFailure(['&ts = [7]'], 'E1012:')
v9.CheckDefExecFailure(['&ts = g:alist'], 'E1012: Type mismatch; expected number but got list<number>')
v9.CheckDefFailure(['&ts = "xx"'], 'E1012:')
v9.CheckDefExecFailure(['&ts = g:astring'], 'E1012: Type mismatch; expected number but got string')
v9.CheckDefFailure(['&path += 3'], 'E1012:')
v9.CheckDefExecFailure(['&bs = "asdf"'], 'E474:')
# test freeing ISN_STOREOPT
v9.CheckDefFailure(['&ts = 3', 'var asdf'], 'E1022:')
&ts = 8
lines =<< trim END
var save_TI = &t_TI
&t_TI = ''
assert_equal('', &t_TI)
&t_TI = 'xxx'
assert_equal('xxx', &t_TI)
&t_TI = save_TI
END
v9.CheckDefAndScriptSuccess(lines)
v9.CheckDefFailure(['&t_TI = 123'], 'E1012:')
v9.CheckScriptFailure(['vim9script', '&t_TI = 123'], 'E928:')
v9.CheckDefFailure(['var s:var = 123'], 'E1101:')
v9.CheckDefFailure(['var s:var: number'], 'E1101:')
v9.CheckDefAndScriptFailure(['var $VAR: number'], ['E1016:', 'E475:'])
lines =<< trim END
vim9script
def SomeFunc()
s:var = 123
enddef
defcompile
END
v9.CheckScriptFailure(lines, 'E1268:')
g:inc_counter += 1
assert_equal(2, g:inc_counter)
var f: float
f += 1
assert_equal(1.0, f)
$SOME_ENV_VAR ..= 'more'
assert_equal('somemore', $SOME_ENV_VAR)
v9.CheckDefFailure(['$SOME_ENV_VAR += "more"'], 'E1051:')
v9.CheckDefFailure(['$SOME_ENV_VAR += 123'], 'E1012:')
v:errmsg = 'none'
v:errmsg ..= 'again'
assert_equal('noneagain', v:errmsg)
v9.CheckDefFailure(['v:errmsg += "more"'], 'E1051:')
v9.CheckDefFailure(['v:errmsg += 123'], 'E1012:')
var text =<< trim END
some text
END
enddef
def Test_float_and_number()
var lines =<< trim END
var f: float
f += 2
f -= 1
assert_equal(1.0, f)
++f
--f
assert_equal(1.0, f)
END
v9.CheckDefAndScriptSuccess(lines)
enddef
let g:someNumber = 43
def Test_assign_concat()
var lines =<< trim END
var s = '-'
s ..= 99
s ..= true
s ..= '-'
s ..= v:null
s ..= g:someNumber
assert_equal('-99true-null43', s)
END
v9.CheckDefAndScriptSuccess(lines)
lines =<< trim END
var s = '-'
s ..= [1, 2]
END
v9.CheckDefAndScriptFailure(lines, ['E1105: Cannot convert list to string', 'E734: Wrong variable type for .='], 2)
lines =<< trim END
var s = '-'
s ..= {a: 2}
END
v9.CheckDefAndScriptFailure(lines, ['E1105: Cannot convert dict to string', 'E734: Wrong variable type for .='], 2)
lines =<< trim END
var ls: list<string> = []
ls[-1] ..= 'foo'
END
v9.CheckDefExecAndScriptFailure(lines, 'E684: List index out of range: -1', 2)
enddef
def Test_assign_register()
var lines =<< trim END
@c = 'areg'
@c ..= 'add'
assert_equal('aregadd', @c)
@@ = 'some text'
assert_equal('some text', getreg('"'))
END
v9.CheckDefAndScriptSuccess(lines)
v9.CheckDefFailure(['@a += "more"'], 'E1051:')
v9.CheckDefFailure(['@a += 123'], 'E1012:')
enddef
def Test_reserved_name()
var more_names = ['null_job', 'null_channel']
if !has('job')
more_names = []
endif
for name in ['true',
'false',
'this',
'super',
'null',
'null_blob',
'null_dict',
'null_function',
'null_list',
'null_partial',
'null_string',
'null_object',
'null_class',
] + more_names
v9.CheckDefExecAndScriptFailure(['var ' .. name .. ' = 0'], 'E1034:')
v9.CheckDefExecAndScriptFailure(['var ' .. name .. ': bool'], 'E1034:')
endfor
var lines =<< trim END
vim9script
def Foo(super: bool)
echo 'something'
enddef
defcompile
END
v9.CheckScriptFailure(lines, 'E1034:')
enddef
def Test_null_values()
var lines =<< trim END
var b: blob = null_blob
var dn: dict<number> = null_dict
var ds: dict<string> = null_dict
var ln: list<number> = null_list
var ls: list<string> = null_list
var Ff: func(string): string = null_function
var Fp: func(number): number = null_partial
var s: string = null_string
if has('job')
var j: job = null_job
var c: channel = null_channel
endif
var d: dict<func> = {a: function('tr'), b: null_function}
var bl: list<blob> = [0z12, null_blob]
var dnl: list<dict<number>> = [{a: 1}, null_dict]
var dsl: list<dict<string>> = [{a: 'x'}, null_dict]
var lnl: list<list<number>> = [[1], null_list]
var lsl: list<list<string>> = [['x'], null_list]
def Len(v: string): number
return len(v)
enddef
var Ffl: list<func(string): number> = [Len, null_function]
var Fpl: list<func(string): number> = [Len, null_partial]
var sl: list<string> = ['x', null_string]
if has('job')
var jl: list<job> = [null_job]
var cl: list<channel> = [null_channel]
endif
END
v9.CheckDefAndScriptSuccess(lines)
enddef
def Test_type_with_extra_white()
var lines =<< trim END
const x : number = 3
END
v9.CheckDefExecAndScriptFailure(lines, 'E1059:')
enddef
def Test_keep_type_after_assigning_null()
var lines =<< trim END
var b: blob
b = null_blob
b = 'text'
END
v9.CheckDefExecAndScriptFailure(lines, 'E1012: Type mismatch; expected blob but got string')
lines =<< trim END
var l: list<number>
l = null_list
l = ['text']
END
v9.CheckDefExecAndScriptFailure(lines, 'E1012: Type mismatch; expected list<number> but got list<string>')
lines =<< trim END
var d: dict<string>
d = null_dict
d = {a: 1, b: 2}
END
v9.CheckDefExecAndScriptFailure(lines, 'E1012: Type mismatch; expected dict<string> but got dict<number>')
enddef
def Test_skipped_assignment()
var lines =<< trim END
for x in []
var i: number = 1
while false
i += 1
endwhile
endfor
END
v9.CheckDefAndScriptSuccess(lines)
enddef
def Test_assign_keep_type()
var lines =<< trim END
vim9script
var l: list<number> = [123]
l = [123]
l->add('string')
END
v9.CheckScriptFailure(lines, 'E1012:', 4)
enddef
def Test_assign_unpack()
var lines =<< trim END
var v1: number
var v2: number
[v1, v2] = [1, 2]
assert_equal(1, v1)
assert_equal(2, v2)
[v1, _, v2, _] = [1, 99, 2, 77]
assert_equal(1, v1)
assert_equal(2, v2)
[v1, v2; _] = [1, 2, 3, 4, 5]
assert_equal(1, v1)
assert_equal(2, v2)
var _x: number
[_x, v2] = [6, 7]
assert_equal(6, _x)
assert_equal(7, v2)
var reslist = []
for text in ['aaa {bbb} ccc', 'ddd {eee} fff']
var before: string
var middle: string
var after: string
[_, before, middle, after; _] = text->matchlist('\(.\{-\}\){\(.\{-\}\)}\(.*\)')
reslist->add(before)->add(middle)->add(after)
endfor
assert_equal(['aaa ', 'bbb', ' ccc', 'ddd ', 'eee', ' fff'], reslist)
var a = 1
var b = 3
[a, b] += [2, 4]
assert_equal(3, a)
assert_equal(7, b)
[a, b] -= [1, 2]
assert_equal(2, a)
assert_equal(5, b)
[a, b] *= [3, 2]
assert_equal(6, a)
assert_equal(10, b)
[a, b] /= [2, 4]
assert_equal(3, a)
assert_equal(2, b)
[a, b] = [17, 15]
[a, b] %= [5, 3]
assert_equal(2, a)
assert_equal(0, b)
END
v9.CheckDefAndScriptSuccess(lines)
lines =<< trim END
var v1: number
var v2: number
[v1, v2] = [1, 2, 3]
END
v9.CheckDefFailure(lines, 'E1093: Expected 2 items but got 3', 3)
lines =<< trim END
var v1: number
var v2: number
[v1, v2] = [1]
END
v9.CheckDefFailure(lines, 'E1093: Expected 2 items but got 1', 3)
lines =<< trim END
var v1: number
var v2: number
[v1, v2; _] = [1]
END
v9.CheckDefFailure(lines, 'E1093: Expected 2 items but got 1', 3)
lines =<< trim END
var v1: number
var v2: number
[v1, v2] =
END
v9.CheckDefFailure(lines, 'E1097:', 5)
lines =<< trim END
var v1: number
var v2: number
[v1, v2] = xxx
END
v9.CheckDefFailure(lines, 'E1001:', 3)
lines =<< trim END
var v1: number
var v2: number
[v1, v2] = popup_clear()
END
v9.CheckDefFailure(lines, 'E1031:', 3)
lines =<< trim END
[v1, v2] = [1, 2]
END
v9.CheckDefFailure(lines, 'E1089:', 1)
v9.CheckScriptFailure(['vim9script'] + lines, 'E1089:', 2)
lines =<< trim END
var v1: number
var v2: number
[v1, v2] = ''
END
v9.CheckDefFailure(lines, 'E1535: List or Tuple required', 3)
lines =<< trim END
g:values = [false, 0]
var x: bool
var y: string
[x, y] = g:values
END
v9.CheckDefExecAndScriptFailure(lines, 'E1163: Variable 2: type mismatch, expected string but got number')
lines =<< trim END
var x: number
var y: number
var z: string
[x, y, z] = [1, 2, 3]
END
v9.CheckDefAndScriptFailure(lines, 'E1163: Variable 3: type mismatch, expected string but got number')
lines =<< trim END
var x: number
var y: string
var z: string
[x, y, z] = [1, '2', 3]
END
v9.CheckDefExecAndScriptFailure(lines, 'E1163: Variable 3: type mismatch, expected string but got number')
enddef
def Test_assign_linebreak()
var nr: number
nr =
123
assert_equal(123, nr)
var n2: number
[nr, n2] =
[12, 34]
assert_equal(12, nr)
assert_equal(34, n2)
v9.CheckDefFailure(["var x = #"], 'E1097:', 3)
var lines =<< trim END
var x: list<string> = ['a']
var y: list<number> = x
->copy()
->copy()
END
v9.CheckDefExecFailure(lines, 'E1012:', 4)
lines =<< trim END
var x: any
x.key = 1
+ 2
+ 3
+ 4
+ 5
END
v9.CheckDefExecAndScriptFailure(lines, ['E1148:', 'E1203:'], 2)
enddef
def Test_assign_index()
# list of list
var l1: list<number>
l1[0] = 123
assert_equal([123], l1)
var l2: list<list<number>>
l2[0] = []
l2[0][0] = 123
assert_equal([[123]], l2)
var l3: list<list<list<number>>>
l3[0] = []
l3[0][0] = []
l3[0][0][0] = 123
assert_equal([[[123]]], l3)
var lines =<< trim END
var l3: list<list<number>>
l3[0] = []
l3[0][0] = []
END
v9.CheckDefFailure(lines, 'E1012: Type mismatch; expected number but got list<any>', 3)
# dict of dict
var d1: dict<number>
d1.one = 1
assert_equal({one: 1}, d1)
var d2: dict<dict<number>>
d2.one = {}
d2.one.two = 123
assert_equal({one: {two: 123}}, d2)
var d3: dict<dict<dict<number>>>
d3.one = {}
d3.one.two = {}
d3.one.two.three = 123
assert_equal({one: {two: {three: 123}}}, d3)
# blob
var bl: blob = 0z11223344
bl[0] = 0x77
assert_equal(0z77223344, bl)
bl[-2] = 0x66
assert_equal(0z77226644, bl)
lines =<< trim END
g:val = '22'
var bl = 0z11
bl[1] = g:val
END
v9.CheckDefExecAndScriptFailure(lines, ['E1030: Using a String as a Number: "22"', 'E1012: Type mismatch; expected number but got string'])
# should not read the next line when generating "a.b"
var a = {}
a.b = {}
a.b.c = {}
->copy()
lines =<< trim END
var d3: dict<dict<number>>
d3.one = {}
d3.one.two = {}
END
v9.CheckDefFailure(lines, 'E1012: Type mismatch; expected number but got dict<any>', 3)
lines =<< trim END
var lines: list<string>
lines['a'] = 'asdf'
END
v9.CheckDefFailure(lines, 'E1012:', 2)
lines =<< trim END
var lines: string
lines[9] = 'asdf'
END
v9.CheckDefFailure(lines, 'E1141:', 2)
# list of dict
var ld: list<dict<number>>
ld[0] = {}
ld[0].one = 123
assert_equal([{one: 123}], ld)
lines =<< trim END
var ld: list<dict<number>>
ld[0] = []
END
v9.CheckDefFailure(lines, 'E1012: Type mismatch; expected dict<number> but got list<any>', 2)
# dict of list
var dl: dict<list<number>>
dl.one = []
dl.one[0] = 123
assert_equal({one: [123]}, dl)
lines =<< trim END
var dl: dict<list<number>>
dl.one = {}
END
v9.CheckDefFailure(lines, 'E1012: Type mismatch; expected list<number> but got dict<any>', 2)
lines =<< trim END
g:l = [1, 2]
g:l['x'] = 3
END
v9.CheckDefExecAndScriptFailure(lines, ['E39:', 'E1030:'], 2)
lines =<< trim END
var bl: blob = test_null_blob()
bl[1] = 8
END
v9.CheckDefExecAndScriptFailure(lines, ['E1184:', 'E979:'], 2)
lines =<< trim END
g:bl = 'not a blob'
g:bl[1 : 2] = 8
END
v9.CheckDefExecAndScriptFailure(lines, ['E897:', 'E689:'], 2)
enddef
def Test_init_in_for_loop()
var lines =<< trim END
var l: list<number> = []
for i in [3, 4]
var n: number
add(l, n)
n = 123
endfor
assert_equal([0, 0], l)
END
v9.CheckDefAndScriptSuccess(lines)
lines =<< trim END
var l: list<number> = []
for i in [3, 4]
var n: number = 0
add(l, n)
n = 123
endfor
assert_equal([0, 0], l)
END
v9.CheckDefAndScriptSuccess(lines)
lines =<< trim END
var l: list<number> = []
for i in [3, 4]
var n: number = 3
add(l, n)
n = 123
endfor
assert_equal([3, 3], l)
END
v9.CheckDefAndScriptSuccess(lines)
enddef
def Test_redir_is_not_assign()
if false
redir => res
echo var_job
redir END
endif
enddef
def Test_extend_list()
# using uninitialized list assigns empty list
var lines =<< trim END
var l1: list<number>
var l2 = l1
assert_true(l1 is l2)
l1 += [123]
assert_equal([123], l1)
assert_true(l1 is l2)
END
v9.CheckDefAndScriptSuccess(lines)
lines =<< trim END
var list: list<string>
extend(list, ['x'])
assert_equal(['x'], list)
END
v9.CheckDefAndScriptSuccess(lines)
# appending to uninitialized list from a function works
lines =<< trim END
vim9script
var list: list<string>
def Func()
list += ['a', 'b']
enddef
Func()
assert_equal(['a', 'b'], list)
END
v9.CheckScriptSuccess(lines)
lines =<< trim END
vim9script
var list: list<string>
def Func()
extend(list, ['x', 'b'])
enddef
Func()
assert_equal(['x', 'b'], list)
END
v9.CheckScriptSuccess(lines)
# initialized to null, with type, does not default to empty list
lines =<< trim END
vim9script
var l: list<string> = test_null_list()
extend(l, ['x'])
END
v9.CheckScriptFailure(lines, 'E1134:', 3)
# initialized to null, without type, does not default to empty list
lines =<< trim END
vim9script
var l = null_list
extend(l, ['x'])
END
v9.CheckScriptFailure(lines, 'E1134:', 3)
# assigned null, does not default to empty list
lines =<< trim END
vim9script
var l: list<string>
l = null_list
extend(l, ['x'])
END
v9.CheckScriptFailure(lines, 'E1134:', 4)
lines =<< trim END
vim9script
extend(test_null_list(), ['x'])
END
v9.CheckScriptFailure(lines, 'E1134:', 2)
# using global var has no declared type
g:myList = []
g:myList->extend([1])
g:myList->extend(['x'])
assert_equal([1, 'x'], g:myList)
unlet g:myList
# using declared list gives an error
lines =<< trim END
var l: list<number>
g:myList = l
g:myList->extend([1])
g:myList->extend(['x'])
END
v9.CheckDefExecAndScriptFailure(lines, 'E1013: Argument 2: type mismatch, expected list<number> but got list<string>', 4)
unlet g:myList
lines =<< trim END
vim9script
var lds = [1, 2, 3]
def Func()
echo lds->extend(['x'])
enddef
defcompile
END
v9.CheckScriptFailure(lines, 'E1013:')
enddef
def Test_extend_dict()
var lines =<< trim END
vim9script
var d: dict<number>
extend(d, {a: 1})
assert_equal({a: 1}, d)
var d2: dict<number>
d2['one'] = 1
assert_equal({one: 1}, d2)
END
v9.CheckScriptSuccess(lines)
lines =<< trim END
vim9script
var d: dict<string> = test_null_dict()
extend(d, {a: 'x'})
END
v9.CheckScriptFailure(lines, 'E1133:', 3)
lines =<< trim END
vim9script
extend(test_null_dict(), {a: 'x'})
END
v9.CheckScriptFailure(lines, 'E1133:', 2)
enddef
def Test_single_letter_vars()
# single letter variables
var a: number = 123
a = 123
assert_equal(123, a)
var b: number
b = 123
assert_equal(123, b)
var g: number
g = 123
assert_equal(123, g)
var s: number
s = 123
assert_equal(123, s)
var t: number
t = 123
assert_equal(123, t)
var v: number
v = 123
assert_equal(123, v)
var w: number
w = 123
assert_equal(123, w)
enddef
def Test_vim9_single_char_vars()
var lines =<< trim END
vim9script
# single character variable declarations work
var a: string
var b: number
var l: list<any>
var s: string
var t: number
var v: number
var w: number
# script-local variables can be used without s: prefix
a = 'script-a'
b = 111
l = [1, 2, 3]
s = 'script-s'
t = 222
v = 333
w = 444
assert_equal('script-a', a)
assert_equal(111, b)
assert_equal([1, 2, 3], l)
assert_equal('script-s', s)
assert_equal(222, t)
assert_equal(333, v)
assert_equal(444, w)
END
writefile(lines, 'Xsinglechar', 'D')
source Xsinglechar
enddef
def Test_assignment_list()
var list1: list<bool> = [false, true, false]
var list2: list<number> = [1, 2, 3]
var list3: list<string> = ['sdf', 'asdf']
var list4: list<any> = ['yes', true, 1234]
var list5: list<blob> = [0z01, 0z02]
var listS: list<string> = []
var listN: list<number> = []
assert_equal([1, 2, 3], list2)
list2[-1] = 99
assert_equal([1, 2, 99], list2)
list2[-2] = 88
assert_equal([1, 88, 99], list2)
list2[-3] = 77
assert_equal([77, 88, 99], list2)
list2 += [100]
assert_equal([77, 88, 99, 100], list2)
list3 += ['end']
assert_equal(['sdf', 'asdf', 'end'], list3)
v9.CheckDefExecFailure(['var ll = [1, 2, 3]', 'll[-4] = 6'], 'E684:')
v9.CheckDefExecFailure(['var ll = [1, 2, 3]', 'unlet ll[8 : 9]'], 'E684:')
v9.CheckDefExecFailure(['var ll = [1, 2, 3]', 'unlet ll[1 : -9]'], 'E684:')
v9.CheckDefExecFailure(['var ll = [1, 2, 3]', 'unlet ll[2 : 1]'], 'E684:')
# type becomes list<any>
var somelist = rand() > 0 ? [1, 2, 3] : ['a', 'b', 'c']
# type is list<any> even though initializer is list<number>
var anyList: list<any> = [0]
assert_equal([0, 'x'], extend(anyList, ['x']))
var lines =<< trim END
var d = {dd: test_null_list()}
d.dd[0] = 0
END
v9.CheckDefExecFailure(lines, 'E1147:', 2)
lines =<< trim END
def OneArg(x: bool)
enddef
def TwoArgs(x: bool, y: bool)
enddef
var fl: list<func(bool, bool, bool)> = [OneArg, TwoArgs]
END
v9.CheckDefExecAndScriptFailure(lines, 'E1012:', 5)
enddef
def Test_list_declaration()
var [v1, v2] = [1, 2]
v1 += 3
assert_equal(4, v1)
v2 *= 3
assert_equal(6, v2)
var lines =<< trim END
var [v1, v2] = [1]
END
v9.CheckDefExecAndScriptFailure(lines, ['E1093: Expected 2 items but got 1', 'E688:'])
lines =<< trim END
var testlist = [1]
var [v1, v2] = testlist
END
v9.CheckDefExecAndScriptFailure(lines, ['E1093: Expected 2 items but got 1', 'E688:'])
lines =<< trim END
var [v1, v2] = [1, 2, 3]
END
v9.CheckDefExecAndScriptFailure(lines, ['E1093: Expected 2 items but got 3', 'E687:'])
lines =<< trim END
var testlist = [1, 2, 3]
var [v1, v2] = testlist
END
v9.CheckDefExecAndScriptFailure(lines, ['E1093: Expected 2 items but got 3', 'E687:'])
var [vnr, vstr] = [123, 'text']
vnr += 3
assert_equal(126, vnr)
vstr ..= 'end'
assert_equal('textend', vstr)
var [vnr2: number, vstr2: string] = [123, 'text']
vnr2 += 3
assert_equal(126, vnr2)
vstr2 ..= 'end'
assert_equal('textend', vstr2)
var [vnr3: number; vlist: list<string>] = [123, 'foo', 'bar']
vnr3 += 5
assert_equal(128, vnr3)
assert_equal(['foo', 'bar'], vlist)
lines =<< trim END
var [vnr2: number, vstr2: number] = [123, 'text']
END
v9.CheckDefExecAndScriptFailure(lines, ['E1163: Variable 2: type mismatch, expected number but got string', 'E1012: Type mismatch; expected number but got string'])
lines =<< trim END
var testlist = [234, 'text']
var [vnr2: number, vstr2: number] = testlist
END
v9.CheckDefExecAndScriptFailure(lines, ['E1163: Variable 2: type mismatch, expected number but got string', 'E1012: Type mismatch; expected number but got string'])
enddef
def PartFuncBool(b: bool): string
return 'done'
enddef
def Test_assignment_partial()
var lines =<< trim END
var Partial: func(): string = function(g:PartFuncBool, [true])
assert_equal('done', Partial())
END
v9.CheckDefAndScriptSuccess(lines)
lines =<< trim END
vim9script
def Func(b: bool)
enddef
var Ref: func = function(Func, [true])
assert_equal('func()', typename(Ref))
Ref()
END
v9.CheckScriptSuccess(lines)
lines =<< trim END
vim9script
var nres: any
var sres: any
def Func(nr: number, s = '')
nres = nr
sres = s
enddef
var n: number
var Ref = function(Func, [n])
Ref('x')
assert_equal(0, nres)
assert_equal('x', sres)
END
v9.CheckScriptSuccess(lines)
lines =<< trim END
vim9script
def Func(nr: number, s = '')
enddef
var n: number
var Ref = function(Func, [n])
Ref(0)
END
v9.CheckScriptFailure(lines, 'E1013: Argument 2: type mismatch, expected string but got number')
lines =<< trim END
var Fn1 = () => {
return 10
}
assert_equal('func(): number', typename(Fn1))
var Fn2 = () => {
return "a"
}
assert_equal('func(): string', typename(Fn2))
var Fn3 = () => {
return {a: [1]}
}
assert_equal('func(): dict<list<number>>', typename(Fn3))
var Fn4 = (...l: list<string>) => {
return []
}
assert_equal('func(...list<string>): list<any>', typename(Fn4))
END
v9.CheckSourceSuccess(['vim9script'] + lines)
v9.CheckSourceSuccess(['def Xfunc()'] + lines + ['enddef', 'defcompile'])
enddef
def Test_assignment_list_any_index()
var l: list<number> = [1, 2]
for [x, y, _]
in [[0, 1, ''], [1, 3, '']]
l[x] = l[x] + y
endfor
assert_equal([2, 5], l)
enddef
def Test_assignment_list_vim9script()
var lines =<< trim END
vim9script
var v1: number
var v2: number
var v3: number
[v1, v2, v3] = [1, 2, 3]
assert_equal([1, 2, 3], [v1, v2, v3])
END
v9.CheckScriptSuccess(lines)
enddef
def Test_assignment_dict()
var dict1: dict<bool> = {one: false, two: true}
var dict2: dict<number> = {one: 1, two: 2}
var dict3: dict<string> = {key: 'value'}
var dict4: dict<any> = {one: 1, two: '2'}
var dict5: dict<blob> = {one: 0z01, two: 0z02}
# check the type is OK
var events: dict<string> = v:event
# overwrite
dict3['key'] = 'another'
assert_equal(dict3, {key: 'another'})
dict3.key = 'yet another'
assert_equal(dict3, {key: 'yet another'})
# member "any" can also be a dict and assigned to
var anydict: dict<any> = {nest: {}, nr: 0}
anydict.nest['this'] = 123
anydict.nest.that = 456
assert_equal({nest: {this: 123, that: 456}, nr: 0}, anydict)
var lines =<< trim END
var dd = {}
dd.two = 2
assert_equal({two: 2}, dd)
END
v9.CheckDefAndScriptSuccess(lines)
lines =<< trim END
var d = {dd: {}}
d.dd[0] = 2
d.dd['x'] = 3
d.dd.y = 4
assert_equal({dd: {0: 2, x: 3, y: 4}}, d)
END
v9.CheckDefAndScriptSuccess(lines)
lines =<< trim END
var key = 'foo'
g:[key] = 'value'
assert_equal('value', g:foo)
unlet g:foo
END
v9.CheckDefAndScriptSuccess(lines)
lines =<< trim END
var dd = {one: 1}
dd.one) = 2
END
v9.CheckDefFailure(lines, 'E488:', 2)
lines =<< trim END
var dd = {one: 1}
var dd.one = 2
END
v9.CheckDefAndScriptFailure(lines, 'E1017:', 2)
# empty key can be used
var dd = {}
dd[""] = 6
assert_equal({['']: 6}, dd)
# type becomes dict<any>
var somedict = rand() > 0 ? {a: 1, b: 2} : {a: 'a', b: 'b'}
# type is dict<any> even though initializer is dict<number>
var anyDict: dict<any> = {a: 0}
assert_equal({a: 0, b: 'x'}, extend(anyDict, {b: 'x'}))
# using global var, which has no declared type
g:myDict = {}
g:myDict->extend({a: 1})
g:myDict->extend({b: 'x'})
assert_equal({a: 1, b: 'x'}, g:myDict)
unlet g:myDict
# using list with declared type gives an error
lines =<< trim END
var d: dict<number>
g:myDict = d
g:myDict->extend({a: 1})
g:myDict->extend({b: 'x'})
END
v9.CheckDefExecAndScriptFailure(lines, 'E1013: Argument 2: type mismatch, expected dict<number> but got dict<string>', 4)
unlet g:myDict
# assignment to script-local dict
lines =<< trim END
vim9script
var test: dict<any> = {}
def FillDict(): dict<any>
test['a'] = 43
return test
enddef
assert_equal({a: 43}, FillDict())
END
v9.CheckScriptSuccess(lines)
lines =<< trim END
vim9script
var test: dict<any>
def FillDict(): dict<any>
test['a'] = 43
return test
enddef
FillDict()
assert_equal({a: 43}, test)
END
v9.CheckScriptSuccess(lines)
# assignment to global dict
lines =<< trim END
vim9script
g:test = {}
def FillDict(): dict<any>
g:test['a'] = 43
return g:test
enddef
assert_equal({a: 43}, FillDict())
END
v9.CheckScriptSuccess(lines)
# assignment to buffer dict
lines =<< trim END
vim9script
b:test = {}
def FillDict(): dict<any>
b:test['a'] = 43
return b:test
enddef
assert_equal({a: 43}, FillDict())
END
v9.CheckScriptSuccess(lines)
lines =<< trim END
var d = {dd: test_null_dict()}
d.dd[0] = 0
END
v9.CheckDefExecFailure(lines, 'E1103:', 2)
lines =<< trim END
var d = {dd: 'string'}
d.dd[0] = 0
END
v9.CheckDefExecFailure(lines, 'E1148:', 2)
lines =<< trim END
var n: any
n.key = 5
END
v9.CheckDefExecAndScriptFailure(lines, ['E1148:', 'E1203: Dot not allowed after a number: n.key = 5'], 2)
enddef
def Test_assignment_local()
# Test in a separated file in order not to the current buffer/window/tab is
# changed.
var script_lines: list<string> =<< trim END
let b:existing = 'yes'
let w:existing = 'yes'
let t:existing = 'yes'
def Test_assignment_local_internal()
b:newvar = 'new'
assert_equal('new', b:newvar)
assert_equal('yes', b:existing)
b:existing = 'no'
assert_equal('no', b:existing)
b:existing ..= 'NO'
assert_equal('noNO', b:existing)
w:newvar = 'new'
assert_equal('new', w:newvar)
assert_equal('yes', w:existing)
w:existing = 'no'
assert_equal('no', w:existing)
w:existing ..= 'NO'
assert_equal('noNO', w:existing)
t:newvar = 'new'
assert_equal('new', t:newvar)
assert_equal('yes', t:existing)
t:existing = 'no'
assert_equal('no', t:existing)
t:existing ..= 'NO'
assert_equal('noNO', t:existing)
enddef
call Test_assignment_local_internal()
END
v9.CheckScriptSuccess(script_lines)
enddef
def Test_assignment_default()
# Test default values.
var thebool: bool
assert_equal(v:false, thebool)
var thenumber: number
assert_equal(0, thenumber)
var thefloat: float
assert_equal(0.0, thefloat)
var thestring: string
assert_equal('', thestring)
var theblob: blob
assert_equal(0z, theblob)
var Thefunc: func
assert_equal(test_null_function(), Thefunc)
var thelist: list<any>
assert_equal([], thelist)
var thedict: dict<any>
assert_equal({}, thedict)
if has('channel')
var thejob: job
assert_equal(test_null_job(), thejob)
var thechannel: channel
assert_equal(test_null_channel(), thechannel)
if has('unix') && executable('cat')
# check with non-null job and channel, types must match
thejob = job_start("cat ", {})
thechannel = job_getchannel(thejob)
job_stop(thejob, 'kill')
endif
endif
var nr = 1234 | nr = 5678
assert_equal(5678, nr)
enddef
def Test_script_var_default()
var lines =<< trim END
vim9script
var l: list<number>
var li = [1, 2]
var bl: blob
var bli = 0z12
var d: dict<number>
var di = {'a': 1, 'b': 2}
def Echo()
assert_equal([], l)
assert_equal([1, 2], li)
assert_equal(0z, bl)
assert_equal(0z12, bli)
assert_equal({}, d)
assert_equal({'a': 1, 'b': 2}, di)
enddef
Echo()
END
v9.CheckScriptSuccess(lines)
enddef
let s:scriptvar = 'init'
def Test_assignment_var_list()
var lines =<< trim END
var v1: string
var v2: string
var vrem: list<string>
[v1] = ['aaa']
assert_equal('aaa', v1)
[v1, v2] = ['one', 'two']
assert_equal('one', v1)
assert_equal('two', v2)
[v1, v2; vrem] = ['one', 'two']
assert_equal('one', v1)
assert_equal('two', v2)
assert_equal([], vrem)
[v1, v2; vrem] = ['one', 'two', 'three']
assert_equal('one', v1)
assert_equal('two', v2)
assert_equal(['three'], vrem)
[&ts, &sw] = [3, 4]
assert_equal(3, &ts)
assert_equal(4, &sw)
set ts=8 sw=4
[@a, @z] = ['aa', 'zz']
assert_equal('aa', @a)
assert_equal('zz', @z)
[$SOME_VAR, $OTHER_VAR] = ['some', 'other']
assert_equal('some', $SOME_VAR)
assert_equal('other', $OTHER_VAR)
[g:globalvar, b:bufvar, w:winvar, t:tabvar, v:errmsg] =
['global', 'buf', 'win', 'tab', 'error']
assert_equal('global', g:globalvar)
assert_equal('buf', b:bufvar)
assert_equal('win', w:winvar)
assert_equal('tab', t:tabvar)
assert_equal('error', v:errmsg)
unlet g:globalvar
END
v9.CheckDefAndScriptSuccess(lines)
[g:globalvar, scriptvar, b:bufvar] = ['global', 'script', 'buf']
assert_equal('global', g:globalvar)
assert_equal('script', scriptvar)
assert_equal('buf', b:bufvar)
lines =<< trim END
vim9script
var scriptvar = 'init'
[g:globalvar, scriptvar, w:winvar] = ['global', 'script', 'win']
assert_equal('global', g:globalvar)
assert_equal('script', scriptvar)
assert_equal('win', w:winvar)
END
v9.CheckScriptSuccess(lines)
enddef
def Test_assignment_empty_list()
var lines =<< trim END
var l2: list<any> = []
var l: list<string>
l = l2
END
v9.CheckDefAndScriptSuccess(lines)
enddef
def Test_assignment_vim9script()
var lines =<< trim END
vim9script
def Func(): list<number>
return [1, 2]
enddef
var name1: number
var name2: number
[name1, name2] =
Func()
assert_equal(1, name1)
assert_equal(2, name2)
var ll =
Func()
assert_equal([1, 2], ll)
@/ = 'text'
assert_equal('text', @/)
@0 = 'zero'
assert_equal('zero', @0)
@1 = 'one'
assert_equal('one', @1)
@9 = 'nine'
assert_equal('nine', @9)
@- = 'minus'
assert_equal('minus', @-)
if has('clipboard_working')
@* = 'star'
assert_equal('star', @*)
@+ = 'plus'
assert_equal('plus', @+)
endif
var a: number = 123
assert_equal(123, a)
var s: string = 'yes'
assert_equal('yes', s)
var b: number = 42
assert_equal(42, b)
var w: number = 43
assert_equal(43, w)
var t: number = 44
assert_equal(44, t)
var to_var = 0
to_var = 3
assert_equal(3, to_var)
END
v9.CheckScriptSuccess(lines)
lines =<< trim END
vim9script
var n: number
def Func()
n = 'string'
enddef
defcompile
END
v9.CheckScriptFailure(lines, 'E1012: Type mismatch; expected number but got string')
enddef
def Mess(): string
v:foldstart = 123
return 'xxx'
enddef
def Test_assignment_failure()
v9.CheckDefFailure(['var name=234'], 'E1004:')
v9.CheckDefFailure(['var name =234'], 'E1004:')
v9.CheckDefFailure(['var name= 234'], 'E1004:')
v9.CheckScriptFailure(['vim9script', 'var name=234'], 'E1004:')
v9.CheckScriptFailure(['vim9script', 'var name=234'], "before and after '='")
v9.CheckScriptFailure(['vim9script', 'var name =234'], 'E1004:')
v9.CheckScriptFailure(['vim9script', 'var name= 234'], 'E1004:')
v9.CheckScriptFailure(['vim9script', 'var name = 234', 'name+=234'], 'E1004:')
v9.CheckScriptFailure(['vim9script', 'var name = 234', 'name+=234'], "before and after '+='")
v9.CheckScriptFailure(['vim9script', 'var name = "x"', 'name..="y"'], 'E1004:')
v9.CheckScriptFailure(['vim9script', 'var name = "x"', 'name..="y"'], "before and after '..='")
v9.CheckDefFailure(['var true = 1'], 'E1034:')
v9.CheckDefFailure(['var false = 1'], 'E1034:')
v9.CheckDefFailure(['var null = 1'], 'E1034:')
v9.CheckDefFailure(['var this = 1'], 'E1034:')
v9.CheckDefFailure(['var super = 1'], 'E1034:')
v9.CheckDefFailure(['[a; b; c] = g:list'], 'E1001:')
v9.CheckDefFailure(['var [a; b; c] = g:list'], 'E1080:')
v9.CheckDefExecFailure(['var a: number',
'[a] = test_null_list()'], 'E1093:')
v9.CheckDefExecFailure(['var a: number',
'[a] = []'], 'E1093:')
v9.CheckDefExecFailure(['var x: number',
'var y: number',
'[x, y] = [1]'], 'E1093:')
v9.CheckDefExecFailure(['var x: string',
'var y: string',
'[x, y] = ["x"]'], 'E1093:')
v9.CheckDefExecFailure(['var x: number',
'var y: number',
'var z: list<number>',
'[x, y; z] = [1]'], 'E1093:')
v9.CheckDefFailure(['var somevar'], "E1022:")
v9.CheckDefFailure(['var &tabstop = 4'], 'E1052:')
v9.CheckDefFailure(['&g:option = 5'], 'E113:')
v9.CheckScriptFailure(['vim9script', 'var &tabstop = 4'], 'E1052:')
v9.CheckDefFailure(['var $VAR = 5'], 'E1016: Cannot declare an environment variable:')
v9.CheckScriptFailure(['vim9script', 'var $ENV = "xxx"'], 'E1016:')
if has('dnd')
v9.CheckDefFailure(['var @~ = 5'], 'E1066:')
else
v9.CheckDefFailure(['var @~ = 5'], 'E354:')
v9.CheckDefFailure(['@~ = 5'], 'E354:')
endif
v9.CheckDefFailure(['var @a = 5'], 'E1066:')
v9.CheckDefFailure(['var @/ = "x"'], 'E1066:')
v9.CheckScriptFailure(['vim9script', 'var @a = "abc"'], 'E1066:')
v9.CheckDefFailure(['var g:var = 5'], 'E1016: Cannot declare a global variable:')
v9.CheckDefFailure(['var w:var = 5'], 'E1016: Cannot declare a window variable:')
v9.CheckDefFailure(['var b:var = 5'], 'E1016: Cannot declare a buffer variable:')
v9.CheckDefFailure(['var t:var = 5'], 'E1016: Cannot declare a tab variable:')
v9.CheckDefFailure(['var anr = 4', 'anr ..= "text"'], 'E1019:')
v9.CheckDefFailure(['var xnr += 4'], 'E1020:', 1)
v9.CheckScriptFailure(['vim9script', 'var xnr += 4'], 'E1020:')
v9.CheckDefFailure(["var xnr = xnr + 1"], 'E1001:', 1)
v9.CheckScriptFailure(['vim9script', 'var xnr = xnr + 4'], 'E121:')
v9.CheckScriptFailure(['vim9script', 'def Func()', 'var dummy = notfound', 'enddef', 'defcompile'], 'E1001:')
v9.CheckDefFailure(['var name: list<string> = [123]'], 'expected list<string> but got list<number>')
v9.CheckDefFailure(['var name: list<number> = ["xx"]'], 'expected list<number> but got list<string>')
v9.CheckDefFailure(['var name: dict<string> = {key: 123}'], 'expected dict<string> but got dict<number>')
v9.CheckDefFailure(['var name: dict<number> = {key: "xx"}'], 'expected dict<number> but got dict<string>')
v9.CheckDefFailure(['var name = feedkeys("0")'], 'E1031:')
v9.CheckDefFailure(['var name: number = feedkeys("0")'], 'expected number but got void')
v9.CheckDefFailure(['var name: dict <number>'], 'E1068:')
v9.CheckDefFailure(['var name: dict<number'], 'E1009: Missing > after type: <number')
assert_fails('s/^/\=g:Mess()/n', 'E794:')
v9.CheckDefFailure(['var name: dict<number'], 'E1009:')
v9.CheckDefFailure(['w:foo: number = 10'],
'E1016: Cannot declare a window variable: w:foo')
v9.CheckDefFailure(['t:foo: bool = true'],
'E1016: Cannot declare a tab variable: t:foo')
v9.CheckDefFailure(['b:foo: string = "x"'],
'E1016: Cannot declare a buffer variable: b:foo')
v9.CheckDefFailure(['g:foo: number = 123'],
'E1016: Cannot declare a global variable: g:foo')
v9.CheckScriptFailure(['vim9script', 'w:foo: number = 123'],
'E1304: Cannot use type with this variable: w:foo:')
v9.CheckScriptFailure(['vim9script', 't:foo: number = 123'],
'E1304: Cannot use type with this variable: t:foo:')
v9.CheckScriptFailure(['vim9script', 'b:foo: number = 123'],
'E1304: Cannot use type with this variable: b:foo:')
v9.CheckScriptFailure(['vim9script', 'g:foo: number = 123'],
'E1304: Cannot use type with this variable: g:foo:')
v9.CheckScriptFailure(['vim9script', 'const w:FOO: number = 123'],
'E1304: Cannot use type with this variable: w:FOO:')
v9.CheckScriptFailure(['vim9script', 'const t:FOO: number = 123'],
'E1304: Cannot use type with this variable: t:FOO:')
v9.CheckScriptFailure(['vim9script', 'const b:FOO: number = 123'],
'E1304: Cannot use type with this variable: b:FOO:')
v9.CheckScriptFailure(['vim9script', 'const g:FOO: number = 123'],
'E1304: Cannot use type with this variable: g:FOO:')
enddef
def Test_assign_list()
var lines =<< trim END
var l: list<string> = []
l[0] = 'value'
assert_equal('value', l[0])
l[1] = 'asdf'
assert_equal('value', l[0])
assert_equal('asdf', l[1])
assert_equal('asdf', l[-1])
assert_equal('value', l[-2])
var nrl: list<number> = []
for i in range(5)
nrl[i] = i
endfor
assert_equal([0, 1, 2, 3, 4], nrl)
var ul: list<any>
ul[0] = 1
ul[1] = 2
ul[2] = 3
assert_equal([1, 2, 3], ul)
END
v9.CheckDefAndScriptSuccess(lines)
lines =<< trim END
var l = [1, 2]
g:idx = 'x'
l[g:idx : 1] = [0]
echo l
END
v9.CheckDefExecAndScriptFailure(lines, ['E1012: Type mismatch; expected number but got string', 'E1030: Using a String as a Number: "x"'])
lines =<< trim END
var l = [1, 2]
g:idx = 3
l[g:idx : 1] = [0]
echo l
END
v9.CheckDefExecAndScriptFailure(lines, 'E684: List index out of range: 3')
lines =<< trim END
var l = [1, 2]
g:idx = 'y'
l[1 : g:idx] = [0]
echo l
END
v9.CheckDefExecAndScriptFailure(lines, ['E1012: Type mismatch; expected number but got string', 'E1030: Using a String as a Number: "y"'])
v9.CheckDefFailure(["var l: list<number> = ['', true]"], 'E1012: Type mismatch; expected list<number> but got list<any>', 1)
v9.CheckDefFailure(["var l: list<list<number>> = [['', true]]"], 'E1012: Type mismatch; expected list<list<number>> but got list<list<any>>', 1)
enddef
def Test_assign_dict()
var lines =<< trim END
var d: dict<string> = {}
d['key'] = 'value'
assert_equal('value', d['key'])
d[123] = 'qwerty'
assert_equal('qwerty', d[123])
assert_equal('qwerty', d['123'])
var nrd: dict<number> = {}
for i in range(3)
nrd[i] = i
endfor
assert_equal({0: 0, 1: 1, 2: 2}, nrd)
d.somekey = 'someval'
assert_equal({key: 'value', '123': 'qwerty', somekey: 'someval'}, d)
unlet d.somekey
assert_equal({key: 'value', '123': 'qwerty'}, d)
END
v9.CheckDefAndScriptSuccess(lines)
v9.CheckDefFailure(["var d: dict<number> = {a: '', b: true}"], 'E1012: Type mismatch; expected dict<number> but got dict<any>', 1)
v9.CheckDefFailure(["var d: dict<dict<number>> = {x: {a: '', b: true}}"], 'E1012: Type mismatch; expected dict<dict<number>> but got dict<dict<any>>', 1)
v9.CheckDefFailure(["var d = {x: 1}", "d[1 : 2] = {y: 2}"], 'E1165: Cannot use a range with an assignment: d[1 : 2] =', 2)
enddef
def Test_assign_dict_unknown_type()
var lines =<< trim END
vim9script
var mylist = []
mylist += [{one: 'one'}]
def Func()
var dd = mylist[0]
assert_equal('one', dd.one)
enddef
Func()
END
v9.CheckScriptSuccess(lines)
lines =<< trim END
vim9script
var mylist = [[]]
mylist[0] += [{one: 'one'}]
def Func()
var dd = mylist[0][0]
assert_equal('one', dd.one)
enddef
Func()
END
v9.CheckScriptSuccess(lines)
enddef
def Test_assign_dict_with_op()
var lines =<< trim END
var ds: dict<string> = {a: 'x'}
ds['a'] ..= 'y'
ds.a ..= 'z'
assert_equal('xyz', ds.a)
var dn: dict<number> = {a: 9}
dn['a'] += 2
assert_equal(11, dn.a)
dn.a += 2
assert_equal(13, dn.a)
dn['a'] -= 3
assert_equal(10, dn.a)
dn.a -= 2
assert_equal(8, dn.a)
dn['a'] *= 2
assert_equal(16, dn.a)
dn.a *= 2
assert_equal(32, dn.a)
dn['a'] /= 3
assert_equal(10, dn.a)
dn.a /= 2
assert_equal(5, dn.a)
dn['a'] %= 3
assert_equal(2, dn.a)
dn.a %= 6
assert_equal(2, dn.a)
var dd: dict<dict<list<any>>>
dd.a = {}
dd.a.b = [0]
dd.a.b += [1]
assert_equal({a: {b: [0, 1]}}, dd)
var dab = {a: ['b']}
dab.a[0] ..= 'c'
assert_equal({a: ['bc']}, dab)
END
v9.CheckDefAndScriptSuccess(lines)
enddef
def Test_assign_list_with_op()
var lines =<< trim END
var ls: list<string> = ['x']
ls[0] ..= 'y'
assert_equal('xy', ls[0])
var ln: list<number> = [9]
ln[0] += 2
assert_equal(11, ln[0])
ln[0] -= 3
assert_equal(8, ln[0])
ln[0] *= 2
assert_equal(16, ln[0])
ln[0] /= 3
assert_equal(5, ln[0])
ln[0] %= 3
assert_equal(2, ln[0])
END
v9.CheckDefAndScriptSuccess(lines)
enddef
def Test_assign_with_op_fails()
var lines =<< trim END
var s = 'abc'
s[1] += 'x'
END
v9.CheckDefAndScriptFailure(lines, ['E1141:', 'E689:'], 2)
lines =<< trim END
var s = 'abc'
s[1] ..= 'x'
END
v9.CheckDefAndScriptFailure(lines, ['E1141:', 'E689:'], 2)
lines =<< trim END
var dd: dict<dict<list<any>>>
dd.a = {}
dd.a.b += [1]
END
v9.CheckDefExecAndScriptFailure(lines, 'E716:', 3)
enddef
def Test_assign_lambda()
# check if assign a lambda to a variable which type is func or any.
var lines =<< trim END
vim9script
var FuncRef = () => 123
assert_equal(123, FuncRef())
var FuncRef_Func: func = () => 123
assert_equal(123, FuncRef_Func())
var FuncRef_Any: any = () => 123
assert_equal(123, FuncRef_Any())
var FuncRef_Number: func(): number = () => 321
assert_equal(321, FuncRef_Number())
END
v9.CheckScriptSuccess(lines)
lines =<< trim END
var Ref: func(number)
Ref = (j) => !j
END
v9.CheckDefAndScriptFailure(lines, 'E1012: Type mismatch; expected func(number) but got func(any): bool')
lines =<< trim END
echo filter([1, 2, 3], (_, v: string) => v + 1)
END
v9.CheckDefAndScriptFailure(lines, 'E1051:')
enddef
def Test_assign_funcref_args()
# unspecified arguments match everything, including varargs
var lines =<< trim END
vim9script
var FuncUnknown: func: number
FuncUnknown = (v): number => v
assert_equal(5, FuncUnknown(5))
FuncUnknown = (v1, v2): number => v1 + v2
assert_equal(7, FuncUnknown(3, 4))
FuncUnknown = (...v1): number => v1[0] + v1[1] + len(v1) * 1000
assert_equal(4007, FuncUnknown(3, 4, 5, 6))
FuncUnknown = (v: list<any>): number => v[0] + v[1] + len(v) * 1000
assert_equal(5009, FuncUnknown([4, 5, 6, 7, 8]))
END
v9.CheckScriptSuccess(lines)
# varargs must match
lines =<< trim END
vim9script
var FuncAnyVA: func(...any): number
FuncAnyVA = (v): number => v
END
v9.CheckScriptFailure(lines, 'E1180: Variable arguments type must be a list: any')
# varargs must match
lines =<< trim END
vim9script
var FuncAnyVA: func(...any): number
FuncAnyVA = (v1, v2): number => v1 + v2
END
v9.CheckScriptFailure(lines, 'E1180: Variable arguments type must be a list: any')
# varargs must match
lines =<< trim END
vim9script
var FuncAnyVA: func(...any): number
FuncAnyVA = (v1: list<any>): number => 3
END
v9.CheckScriptFailure(lines, 'E1180: Variable arguments type must be a list: any')
enddef
def Test_assign_funcref_arg_any()
var lines =<< trim END
vim9script
var FuncAnyVA: func(any): number
FuncAnyVA = (v): number => v
END
# TODO: Verify this should succeed.
v9.CheckScriptSuccess(lines)
enddef
def Test_heredoc()
# simple heredoc
var lines =<< trim END
var text =<< trim TEXT # comment
abc
TEXT
assert_equal(['abc'], text)
END
v9.CheckDefAndScriptSuccess(lines)
# empty heredoc
lines =<< trim END
var text =<< trim TEXT
TEXT
assert_equal([], text)
END
v9.CheckDefAndScriptSuccess(lines)
# heredoc with a single empty line
lines =<< trim END
var text =<< trim TEXT
TEXT
assert_equal([''], text)
END
v9.CheckDefAndScriptSuccess(lines)
# assign heredoc to variable with type
lines =<< trim END
var text: list<string> =<< trim TEXT
var foo =<< trim FOO
TEXT
assert_equal(['var foo =<< trim FOO'], text)
END
v9.CheckDefAndScriptSuccess(lines)
# extra whitespace before type is allowed
lines =<< trim END
var text: list<string> =<< trim TEXT
var foo =<< trim FOO
TEXT
assert_equal(['var foo =<< trim FOO'], text)
END
v9.CheckDefAndScriptSuccess(lines)
# missing whitespace before type is an error
lines =<< trim END
var text:list<string> =<< trim TEXT
var foo =<< trim FOO
TEXT
assert_equal(['var foo =<< trim FOO'], text)
END
v9.CheckDefAndScriptFailure(lines, 'E1069:')
# assign heredoc to list slice
lines =<< trim END
var text = ['']
text[ : ] =<< trim TEXT
var foo =<< trim FOO
TEXT
assert_equal(['var foo =<< trim FOO'], text)
END
v9.CheckDefAndScriptSuccess(lines)
# assign heredoc to curly braces name in legacy function in Vim9 script
lines =<< trim END
vim9script
func Func()
let foo_3_bar = ['']
let foo_{1 + 2}_bar[ : ] =<< trim TEXT
var foo =<< trim FOO
TEXT
call assert_equal(['var foo =<< trim FOO'], foo_3_bar)
endfunc
Func()
END
v9.CheckScriptSuccess(lines)
# commented out heredoc assignment without space after '#'
lines =<< trim END
vim9script
def Func()
#x =<< trim [CODE]
#[CODE]
enddef
Func()
END
v9.CheckScriptSuccess(lines)
# heredoc start should not be recognized in string
lines =<< trim END
vim9script
def Func()
new
@" = 'bar'
['foo', @"]->setline("]=<<"->count('='))
assert_equal(['foo', 'bar'], getline(1, '$'))
bwipe!
enddef
Func()
END
v9.CheckScriptSuccess(lines)
v9.CheckDefFailure(['var lines =<< trim END X', 'END'], 'E488:')
v9.CheckDefFailure(['var lines =<< trim END " comment', 'END'], 'E488:')
lines =<< trim [END]
def Func()
var&lines =<< trim END
x
x
enddef
defcompile
[END]
v9.CheckScriptFailure(lines, 'E1145: Missing heredoc end marker: END')
delfunc! g:Func
lines =<< trim [END]
def Func()
var lines =<< trim END
x
x
x
x
x
x
x
x
enddef
call Func()
[END]
v9.CheckScriptFailure(lines, 'E1145: Missing heredoc end marker: END')
delfunc! g:Func
lines =<< trim END
var lines: number =<< trim STOP
aaa
bbb
STOP
END
v9.CheckDefAndScriptFailure(lines, 'E1012: Type mismatch; expected number but got list<string>', 1)
lines =<< trim END
var lines=<< STOP
xxx
STOP
END
v9.CheckDefAndScriptFailure(lines, 'E1004: White space required before and after ''=<<'' at "=<< STOP"', 1)
lines =<< trim END
var lines =<<STOP
xxx
STOP
END
v9.CheckDefAndScriptFailure(lines, 'E1004: White space required before and after ''=<<'' at "=<<STOP"', 1)
lines =<< trim END
var lines=<<STOP
xxx
STOP
END
v9.CheckDefAndScriptFailure(lines, 'E1004: White space required before and after ''=<<'' at "=<<STOP"', 1)
enddef
def Test_var_func_call()
var lines =<< trim END
vim9script
func GetValue()
if exists('g:count')
let g:count += 1
else
let g:count = 1
endif
return 'this'
endfunc
var val: string = GetValue()
# env var is always a string
var env = $TERM
END
writefile(lines, 'Xfinished', 'D')
source Xfinished
# GetValue() is not called during discovery phase
assert_equal(1, g:count)
unlet g:count
enddef
def Test_var_missing_type()
var lines =<< trim END
vim9script
var name = g:unknown
END
v9.CheckScriptFailure(lines, 'E121:')
lines =<< trim END
vim9script
var nr: number = 123
var name = nr
END
v9.CheckScriptSuccess(lines)
enddef
def Test_var_declaration()
var lines =<< trim END
vim9script
var name: string
g:var_uninit = name
name = 'text'
g:var_test = name
# prefixing s: is not allowed
name = 'prefixed'
g:var_prefixed = name
const FOO: number = 123
assert_equal(123, FOO)
const FOOS = 'foos'
assert_equal('foos', FOOS)
final FLIST = [1]
assert_equal([1], FLIST)
FLIST[0] = 11
assert_equal([11], FLIST)
const g:FOOS = 'gfoos'
assert_equal('gfoos', g:FOOS)
final g:FLIST = [2]
assert_equal([2], g:FLIST)
g:FLIST[0] = 22
assert_equal([22], g:FLIST)
def SetGlobalConst()
const g:globConst = 123
enddef
SetGlobalConst()
assert_equal(123, g:globConst)
assert_true(islocked('g:globConst'))
const w:FOOS = 'wfoos'
assert_equal('wfoos', w:FOOS)
final w:FLIST = [3]
assert_equal([3], w:FLIST)
w:FLIST[0] = 33
assert_equal([33], w:FLIST)
var s:other: number
other = 1234
g:other_var = other
var xyz: string # comment
# type is inferred
var dict = {['a']: 222}
def GetDictVal(key: any)
g:dict_val = dict[key]
enddef
GetDictVal('a')
final adict: dict<string> = {}
def ChangeAdict()
adict.foo = 'foo'
enddef
ChangeAdict()
END
v9.CheckScriptSuccess(lines)
assert_equal('', g:var_uninit)
assert_equal('text', g:var_test)
assert_equal('prefixed', g:var_prefixed)
assert_equal(1234, g:other_var)
assert_equal(222, g:dict_val)
unlet g:var_uninit
unlet g:var_test
unlet g:var_prefixed
unlet g:other_var
unlet g:globConst
unlet g:FOOS
unlet g:FLIST
unlet w:FOOS
unlet w:FLIST
enddef
def Test_create_list_after_const()
const a = 1
g:ll = []
assert_equal(0, islocked('g:ll'))
unlet g:ll
enddef
def Test_var_declaration_fails()
var lines =<< trim END
vim9script
final var: string
END
v9.CheckScriptFailure(lines, 'E1125:')
lines =<< trim END
vim9script
const g:constvar = 'string'
g:constvar = 'xx'
END
v9.CheckScriptFailure(lines, 'E741:')
unlet g:constvar
lines =<< trim END
vim9script
var name = 'one'
lockvar name
def SetLocked()
name = 'two'
enddef
SetLocked()
END
v9.CheckScriptFailure(lines, 'E741: Value is locked: name', 1)
lines =<< trim END
let s:legacy = 'one'
lockvar s:legacy
def SetLocked()
s:legacy = 'two'
enddef
call SetLocked()
END
v9.CheckScriptFailure(lines, 'E741: Value is locked: s:legacy', 1)
lines =<< trim END
vim9script
def SetGlobalConst()
const g:globConst = 123
enddef
SetGlobalConst()
g:globConst = 234
END
v9.CheckScriptFailure(lines, 'E741: Value is locked: g:globConst', 6)
unlet g:globConst
lines =<< trim END
vim9script
const cdict: dict<string> = {}
def Change()
cdict.foo = 'foo'
enddef
defcompile
END
v9.CheckScriptFailure(lines, 'E46:')
lines =<< trim END
vim9script
final w:finalvar = [9]
w:finalvar = [8]
END
v9.CheckScriptFailure(lines, 'E1122:')
unlet w:finalvar
lines =<< trim END
vim9script
const var: string
END
v9.CheckScriptFailure(lines, 'E1021:')
lines =<< trim END
vim9script
var 9var: string
END
v9.CheckScriptFailure(lines, 'E488:')
v9.CheckDefFailure(['var foo.bar = 2'], 'E1087:')
v9.CheckDefFailure(['var foo[3] = 2'], 'E1087:')
v9.CheckDefFailure(['const foo: number'], 'E1021:')
lines =<< trim END
va foo = 123
END
v9.CheckDefAndScriptFailure(lines, 'E1065:', 1)
lines =<< trim END
var foo: func(number
END
v9.CheckDefAndScriptFailure(lines, 'E110:', 1)
lines =<< trim END
var foo: func(number): func(
END
v9.CheckDefAndScriptFailure(lines, 'E110:', 1)
for type in ['num_ber',
'anys', 'ani',
'bools', 'boel',
'blobs', 'blub',
'channels', 'channol',
'dicts', 'duct',
'floats', 'floot',
'funcs', 'funk',
'jobs', 'jop',
'lists', 'last',
'numbers', 'numbar',
'strings', 'strung',
'voids', 'viod']
v9.CheckDefAndScriptFailure([$'var foo: {type}'], 'E1010:', 1)
endfor
enddef
def Test_var_declaration_inferred()
# check that type is set on the list so that extend() fails
var lines =<< trim END
vim9script
def GetList(): list<number>
var l = [1, 2, 3]
return l
enddef
echo GetList()->extend(['x'])
END
v9.CheckScriptFailure(lines, 'E1013:', 6)
lines =<< trim END
vim9script
def GetNr(): number
return 5
enddef
def TestOne()
var some = [function('len'), GetNr]
g:res = typename(some)
enddef
TestOne()
assert_equal('list<func(): number>', g:res)
def TestTwo()
var some = [function('len'), GetNr]
g:res = typename(some)
enddef
TestTwo()
assert_equal('list<func(): number>', g:res)
unlet g:res
# FIXME: why is the type different?
var first = [function('len'), GetNr]
assert_equal('list<func(...): number>', typename(first))
var second = [GetNr, function('len')]
assert_equal('list<func(...): number>', typename(second))
END
v9.CheckScriptSuccess(lines)
enddef
def Test_script_local_in_legacy()
# OK to define script-local later but before compiling
var lines =<< trim END
def SetLater()
legvar = 'two'
enddef
let s:legvar = 'one'
defcompile
call SetLater()
call assert_equal('two', s:legvar)
END
v9.CheckScriptSuccess(lines)
# OK to leave out s: prefix when script-local already defined
lines =<< trim END
let s:legvar = 'one'
def SetNoPrefix()
legvar = 'two'
enddef
call SetNoPrefix()
call assert_equal('two', s:legvar)
END
v9.CheckScriptSuccess(lines)
# Not OK to leave out s: prefix when script-local defined after compiling
lines =<< trim END
def SetLaterNoPrefix()
legvar = 'two'
enddef
defcompile
let s:legvar = 'one'
END
v9.CheckScriptFailure(lines, 'E476:', 1)
edit! Xslfile
lines =<< trim END
var edit: bool
legacy edit
END
v9.CheckDefAndScriptSuccess(lines)
enddef
def Test_var_type_check()
var lines =<< trim END
vim9script
var name: string
name = 1234
END
v9.CheckScriptFailure(lines, 'E1012:')
lines =<< trim END
vim9script
var name:string
END
v9.CheckScriptFailure(lines, 'E1069:')
v9.CheckDefAndScriptFailure(['var n:number = 42'], 'E1069:')
lines =<< trim END
vim9script
var name: asdf
END
v9.CheckScriptFailure(lines, 'E1010:')
lines =<< trim END
vim9script
var l: list<number>
l = []
END
v9.CheckScriptSuccess(lines)
lines =<< trim END
vim9script
var d: dict<number>
d = {}
END
v9.CheckScriptSuccess(lines)
lines =<< trim END
vim9script
var d = {a: 1, b: [2]}
def Func(b: bool)
var l: list<number> = b ? d.b : [3]
enddef
defcompile
END
v9.CheckScriptSuccess(lines)
enddef
let g:dict_number = #{one: 1, two: 2}
def Test_var_list_dict_type()
var ll: list<number>
ll = [1, 2, 2, 3, 3, 3]->uniq()
ll->assert_equal([1, 2, 3])
var dd: dict<number>
dd = g:dict_number
dd->assert_equal(g:dict_number)
var lines =<< trim END
var ll: list<number>
ll = [1, 2, 3]->map('"one"')
END
v9.CheckDefExecFailure(lines, 'E1012: Type mismatch; expected list<number> but got list<string>')
enddef
def Test_cannot_use_let()
v9.CheckDefAndScriptFailure(['let a = 34'], 'E1126:', 1)
enddef
def Test_unlet()
g:somevar = 'yes'
assert_true(exists('g:somevar'))
unlet g:somevar
assert_false(exists('g:somevar'))
unlet! g:somevar
# script-local variable cannot be removed in Vim9 script
s:somevar = 'local'
assert_true(exists('s:somevar'))
v9.CheckDefExecFailure(['unlet s:somevar'], 'E1081:', 1)
v9.CheckDefExecFailure(['unlet! s:somevar'], 'E1081:', 1)
if 0
unlet g:does_not_exist
endif
v9.CheckDefExecFailure(['unlet v:notfound.key'], 'E1001:')
v9.CheckDefExecFailure([
'var dd = 111',
'unlet dd',
], 'E1081:', 2)
# dict unlet
var dd = {a: 1, b: 2, c: 3, 4: 4}
unlet dd['a']
unlet dd.c
unlet dd[4]
assert_equal({b: 2}, dd)
# null key works like empty string
dd = {'': 1, x: 9}
unlet dd[null_string]
assert_equal({x: 9}, dd)
# list unlet
var ll = [1, 2, 3, 4]
unlet ll[1]
unlet ll[-1]
assert_equal([1, 3], ll)
ll = [1, 2, 3, 4]
unlet ll[0 : 1]
assert_equal([3, 4], ll)
ll = [1, 2, 3, 4]
unlet ll[2 : 8]
assert_equal([1, 2], ll)
ll = [1, 2, 3, 4]
unlet ll[-2 : -1]
assert_equal([1, 2], ll)
g:nrdict = {1: 1, 2: 2}
g:idx = 1
unlet g:nrdict[g:idx]
assert_equal({2: 2}, g:nrdict)
unlet g:nrdict
unlet g:idx
v9.CheckDefFailure([
'var ll = [1, 2]',
'll[1 : 2] = 7',
], 'E1012: Type mismatch; expected list<number> but got number', 2)
v9.CheckDefFailure([
'var dd = {a: 1}',
'unlet dd["a" : "a"]',
], 'E1166:', 2)
v9.CheckDefExecFailure([
'unlet g:adict[0 : 1]',
], 'E1148:', 1)
v9.CheckDefFailure([
'var ll = [1, 2]',
'unlet ll[0:1]',
], 'E1004:', 2)
v9.CheckDefFailure([
'var ll = [1, 2]',
'unlet ll[0 :1]',
], 'E1004:', 2)
v9.CheckDefFailure([
'var ll = [1, 2]',
'unlet ll[0: 1]',
], 'E1004:', 2)
v9.CheckDefExecFailure([
'g:ll = [1, 2]',
'g:idx = "x"',
'unlet g:ll[g:idx]',
], 'E1029: Expected number but got string', 3)
v9.CheckDefExecFailure([
'g:ll = [1, 2, 3]',
'g:idx = "x"',
'unlet g:ll[g:idx : 2]',
], 'E1029: Expected number but got string', 3)
v9.CheckDefExecFailure([
'g:ll = [1, 2, 3]',
'g:idx = "x"',
'unlet g:ll[0 : g:idx]',
], 'E1029: Expected number but got string', 3)
# command recognized as assignment when skipping, should not give an error
v9.CheckScriptSuccess([
'vim9script',
'for i in []',
" put =''",
'endfor'])
v9.CheckDefFailure([
'var ll = [1, 2]',
'unlet ll["x" : 1]',
], 'E1012:', 2)
v9.CheckDefFailure([
'var ll = [1, 2]',
'unlet ll[0 : "x"]',
], 'E1012:', 2)
# list of dict unlet
var dl = [{a: 1, b: 2}, {c: 3}]
unlet dl[0]['b']
assert_equal([{a: 1}, {c: 3}], dl)
v9.CheckDefExecFailure([
'var ll = test_null_list()',
'unlet ll[0]',
], 'E684:', 2)
v9.CheckDefExecFailure([
'var ll = [1]',
'unlet ll[2]',
], 'E684:', 2)
v9.CheckDefExecFailure([
'var ll = [1]',
'unlet ll[g:astring]',
], 'E1012:', 2)
v9.CheckDefExecFailure([
'var dd = test_null_dict()',
'unlet dd["a"]',
], 'E716:', 2)
v9.CheckDefExecFailure([
'var dd = {a: 1}',
'unlet dd["b"]',
], 'E716:', 2)
v9.CheckDefExecFailure([
'var dd = {a: 1}',
'unlet dd[g:alist]',
], 'E1105:', 2)
v9.CheckDefExecFailure([
'g:dd = {"a": 1, 2: 2}',
'unlet g:dd[0z11]',
], 'E1029:', 2)
v9.CheckDefExecFailure([
'g:str = "a string"',
'unlet g:str[0]',
], 'E1148: Cannot index a string', 2)
# can compile unlet before variable exists
g:someDict = {key: 'val'}
var k = 'key'
unlet g:someDict[k]
assert_equal({}, g:someDict)
unlet g:someDict
assert_false(exists('g:someDict'))
v9.CheckScriptFailure([
'vim9script',
'var svar = 123',
'unlet svar',
], 'E1081:')
v9.CheckScriptFailure([
'vim9script',
'var svar = 123',
'unlet s:svar',
], 'E1268:')
v9.CheckScriptFailure([
'vim9script',
'var svar = 123',
'def Func()',
' unlet svar',
'enddef',
'defcompile',
], 'E1081:')
v9.CheckScriptSuccess([
'vim9script',
'var svar = 123',
'func Func()',
' unlet s:svar',
'endfunc',
'Func()',
])
v9.CheckScriptFailure([
'vim9script',
'var svar = 123',
'def Func()',
' vim9cmd unlet s:svar',
'enddef',
'defcompile',
], 'E1081:')
v9.CheckScriptFailure([
'vim9script',
'var svar = 123',
'def Func()',
' unlet s:svar',
'enddef',
'defcompile',
], 'E1081:')
v9.CheckScriptFailure([
'vim9script',
'def Delcount(dict: dict<any>)',
' unlet dict.count',
'enddef',
'Delcount(v:)',
], 'E742:')
v9.CheckScriptFailure([
'vim9script',
'def DelChangedtick(dict: dict<any>)',
' unlet dict.changedtick',
'enddef',
'DelChangedtick(b:)',
], 'E795:')
writefile(['vim9script', 'export var svar = 1234'], 'XunletExport.vim', 'D')
var lines =<< trim END
vim9script
import './XunletExport.vim' as exp
def UnletSvar()
unlet exp.svar
enddef
defcompile
END
v9.CheckScriptFailure(lines, 'E1260:', 1)
$ENVVAR = 'foobar'
assert_equal('foobar', $ENVVAR)
unlet $ENVVAR
assert_equal('', $ENVVAR)
enddef
def Test_expr_error_no_assign()
var lines =<< trim END
vim9script
var x = invalid
echo x
END
v9.CheckScriptFailureList(lines, ['E121:', 'E121:'])
lines =<< trim END
vim9script
var x = 1 / 0
echo x
END
v9.CheckScriptFailure(lines, 'E1154:')
lines =<< trim END
vim9script
var x = 1 % 0
echo x
END
v9.CheckScriptFailure(lines, 'E1154:')
lines =<< trim END
var x: string 'string'
END
v9.CheckDefAndScriptFailure(lines, 'E488:')
enddef
def Test_assign_command_modifier()
var lines =<< trim END
var verbose = 0
verbose = 1
assert_equal(1, verbose)
silent verbose = 2
assert_equal(2, verbose)
silent verbose += 2
assert_equal(4, verbose)
silent verbose -= 1
assert_equal(3, verbose)
var topleft = {one: 1}
sandbox topleft.one = 3
assert_equal({one: 3}, topleft)
leftabove topleft[' '] = 4
assert_equal({one: 3, ' ': 4}, topleft)
var x: number
var y: number
silent [x, y] = [1, 2]
assert_equal(1, x)
assert_equal(2, y)
END
v9.CheckDefAndScriptSuccess(lines)
enddef
def Test_assign_alt_buf_register()
var lines =<< trim END
edit 'file_b1'
var b1 = bufnr()
edit 'file_b2'
var b2 = bufnr()
assert_equal(b1, bufnr('#'))
@# = b2
assert_equal(b2, bufnr('#'))
END
v9.CheckDefAndScriptSuccess(lines)
enddef
def Test_script_funcref_case()
var lines =<< trim END
var Len = (s: string): number => len(s) + 1
assert_equal(5, Len('asdf'))
END
v9.CheckDefAndScriptSuccess(lines)
lines =<< trim END
var len = (s: string): number => len(s) + 1
END
v9.CheckDefAndScriptFailure(lines, 'E704:')
lines =<< trim END
vim9script
var Len = (s: string): number => len(s) + 2
assert_equal(6, Len('asdf'))
END
v9.CheckScriptSuccess(lines)
lines =<< trim END
vim9script
var len = (s: string): number => len(s) + 1
END
v9.CheckScriptFailure(lines, 'E704:')
enddef
def Test_script_funcref_runtime_type_check()
var lines =<< trim END
vim9script
def FuncWithNumberArg(n: number)
enddef
def Test()
var Ref: func(string) = function(FuncWithNumberArg)
enddef
defcompile
END
# OK at compile time
v9.CheckScriptSuccess(lines)
# Type check fails at runtime
v9.CheckScriptFailure(lines + ['Test()'], 'E1012: Type mismatch; expected func(string) but got func(number)')
enddef
def Test_inc_dec()
var lines =<< trim END
var nr = 7
++nr
assert_equal(8, nr)
--nr
assert_equal(7, nr)
++nr | ++nr
assert_equal(9, nr)
++nr # comment
assert_equal(10, nr)
var ll = [1, 2]
--ll[0]
++ll[1]
assert_equal([0, 3], ll)
g:count = 1
++g:count
--g:count
assert_equal(1, g:count)
unlet g:count
END
v9.CheckDefAndScriptSuccess(lines)
lines =<< trim END
var nr = 7
++ nr
END
v9.CheckDefAndScriptFailure(lines, "E1202: No white space allowed after '++': ++ nr")
enddef
def Test_abort_after_error()
# should abort after strpart() fails, not give another type error
var lines =<< trim END
vim9script
var x: string
x = strpart(1, 2)
END
writefile(lines, 'Xtestscript', 'D')
var expected = 'E1174: String required for argument 1'
assert_fails('so Xtestscript', [expected, expected], 3)
enddef
def Test_using_s_var_in_function()
var lines =<< trim END
vim9script
var scriptlevel = 123
def SomeFunc()
echo s:scriptlevel
enddef
SomeFunc()
END
v9.CheckScriptFailure(lines, 'E1268:')
# OK in legacy script
lines =<< trim END
let s:scriptlevel = 123
def s:SomeFunc()
echo s:scriptlevel
enddef
call s:SomeFunc()
END
v9.CheckScriptSuccess(lines)
lines =<< trim END
vim9script
var scriptlevel = 123
def SomeFunc()
s:scriptlevel = 456
enddef
SomeFunc()
END
v9.CheckScriptFailure(lines, 'E1268:')
# OK in legacy script
lines =<< trim END
let s:scriptlevel = 123
def s:SomeFunc()
s:scriptlevel = 456
enddef
call s:SomeFunc()
call assert_equal(456, s:scriptlevel)
END
v9.CheckScriptSuccess(lines)
enddef
" Test for specifying a type in assignment
def Test_type_specification_in_assignment()
# specify type for an existing script local variable without "var"
var lines =<< trim END
vim9script
var n: number = 10
n: number = 20
END
v9.CheckSourceFailure(lines, 'E488: Trailing characters: : number = 20', 3)
# specify type for a non-existing script local variable without "var"
lines =<< trim END
vim9script
MyVar: string = 'abc'
END
v9.CheckSourceFailure(lines, "E492: Not an editor command: MyVar: string = 'abc'", 2)
# specify type for an existing def local variable without "var"
lines =<< trim END
vim9script
def Foo()
var n: number = 10
n: number = 20
enddef
Foo()
END
v9.CheckSourceFailure(lines, 'E488: Trailing characters: : number = 20', 2)
# specify type for a non-existing def local variable without "var"
lines =<< trim END
vim9script
def Foo()
MyVar: string = 'abc'
enddef
Foo()
END
v9.CheckSourceFailure(lines, "E476: Invalid command: MyVar: string = 'abc'", 1)
enddef
let g:someVar = 'X'
" Test for heredoc with Vim expressions.
" This messes up highlighting, keep it near the end.
def Test_heredoc_expr()
var lines =<< trim CODE
var s = "local"
var a1 = "1"
var a2 = "2"
var a3 = "3"
var a4 = ""
var code =<< trim eval END
var a = {5 + 10}
var b = {min([10, 6])} + {max([4, 6])}
var c = "{s}"
var d = x{a1}x{a2}x{a3}x{a4}
END
assert_equal(['var a = 15', 'var b = 6 + 6', 'var c = "local"', 'var d = x1x2x3x'], code)
CODE
v9.CheckDefAndScriptSuccess(lines)
# Evaluate a dictionary
lines =<< trim CODE
var d1 = {'a': 10, 'b': [1, 2]}
var code =<< trim eval END
var d2 = {d1}
END
assert_equal(["var d2 = {'a': 10, 'b': [1, 2]}"], code)
CODE
v9.CheckDefAndScriptSuccess(lines)
# Evaluate an empty dictionary
lines =<< trim CODE
var d1 = {}
var code =<< trim eval END
var d2 = {d1}
END
assert_equal(["var d2 = {}"], code)
CODE
v9.CheckDefAndScriptSuccess(lines)
# Evaluate a null dictionary
lines =<< trim CODE
var d1 = test_null_dict()
var code =<< trim eval END
var d2 = {d1}
END
assert_equal(["var d2 = {}"], code)
CODE
v9.CheckDefAndScriptSuccess(lines)
# Evaluate a List
lines =<< trim CODE
var l1 = ['a', 'b', 'c']
var code =<< trim eval END
var l2 = {l1}
END
assert_equal(["var l2 = ['a', 'b', 'c']"], code)
CODE
v9.CheckDefAndScriptSuccess(lines)
# Evaluate an empty List
lines =<< trim CODE
var l1 = []
var code =<< trim eval END
var l2 = {l1}
END
assert_equal(["var l2 = []"], code)
CODE
v9.CheckDefAndScriptSuccess(lines)
# Evaluate a null List
lines =<< trim CODE
var l1 = test_null_list()
var code =<< trim eval END
var l2 = {l1}
END
assert_equal(["var l2 = []"], code)
CODE
v9.CheckDefAndScriptSuccess(lines)
lines =<< trim CODE
var code =<< eval trim END
var s = "{$SOME_ENV_VAR}"
END
assert_equal(['var s = "somemore"'], code)
CODE
v9.CheckDefAndScriptSuccess(lines)
lines =<< trim CODE
var code =<< eval END
var s = "{$SOME_ENV_VAR}"
END
assert_equal([' var s = "somemore"'], code)
CODE
v9.CheckDefAndScriptSuccess(lines)
lines =<< trim CODE
var code =<< eval trim END
let a = {{abc}}
let b = {g:someVar}
let c = {{
END
assert_equal(['let a = {abc}', 'let b = X', 'let c = {'], code)
CODE
v9.CheckDefAndScriptSuccess(lines)
lines =<< trim LINES
var text =<< eval trim END
let b = {
END
LINES
v9.CheckDefAndScriptFailure(lines, "E1279: Missing '}'")
lines =<< trim LINES
var text =<< eval trim END
let b = {abc
END
LINES
v9.CheckDefAndScriptFailure(lines, "E1279: Missing '}'")
lines =<< trim LINES
var text =<< eval trim END
let b = {}
END
LINES
v9.CheckDefAndScriptFailure(lines, 'E15: Invalid expression: "}"')
# dangling "}"
lines =<< trim LINES
var text =<< trim eval END
aa}a
END
LINES
v9.CheckDefAndScriptFailure(lines, "E1278: Stray '}' without a matching '{': aa}a")
enddef
" Test for assigning to a multi-dimensional list item.
def Test_list_item_assign()
var lines =<< trim END
vim9script
def Foo()
var l: list<list<string>> = [['x', 'x', 'x'], ['y', 'y', 'y']]
var z: number = 1
[l[1][2], z] = ['a', 20]
assert_equal([['x', 'x', 'x'], ['y', 'y', 'a']], l)
enddef
Foo()
END
v9.CheckSourceSuccess(lines)
lines =<< trim END
vim9script
var l: list<list<string>> = [['x', 'x', 'x'], ['y', 'y', 'y']]
var z: number = 1
[l[1][2], z] = ['a', 20]
assert_equal([['x', 'x', 'x'], ['y', 'y', 'a']], l)
END
v9.CheckSourceSuccess(lines)
enddef
" Test for assigning to a multi-dimensional dict item.
def Test_dict_item_assign()
# This used to fail with the error "E1105: Cannot convert list to string"
# (Github issue #13485)
var lines =<< trim END
vim9script
def F()
var d: dict<dict<number>> = {a: {b: 0}}
for group in keys(d)
d['a']['b'] += 1
endfor
assert_equal({a: {b: 1}}, d)
enddef
F()
END
v9.CheckSourceSuccess(lines)
# This used to crash Vim
lines =<< trim END
vim9script
def F()
var d: dict<dict<number>> = {a: {b: 0}}
d['a']['b'] += 1
assert_equal({a: {b: 1}}, d)
enddef
F()
END
v9.CheckSourceSuccess(lines)
# Assignment at script level
lines =<< trim END
vim9script
var d: dict<dict<number>> = {a: {b: 0}}
for group in keys(d)
d['a']['b'] += 1
endfor
assert_equal({a: {b: 1}}, d)
END
v9.CheckSourceSuccess(lines)
enddef
def Test_class_assign()
var lines =<< trim END
vim9script
class C
endclass
class D
endclass
assert_fails('C = D', 'E1405: Class "D" cannot be used as a value')
END
v9.CheckSourceSuccess(lines)
enddef
" Test for using various types (dict, list, blob, funcref, class) as variable
" in assignments with a different type
def Test_type_check()
var lines =<< trim END
vim9script
class A
endclass
type T = number
var N: number = 1
var S: string = 'abc'
var d: dict<number> = {}
var l: list<number> = []
var b: blob = 0z10
var Fn: func = function('min')
var o: A = A.new()
# Assign a number
assert_fails('d = N', 'E1012: Type mismatch; expected dict<number> but got number')
assert_fails('l = N', 'E1012: Type mismatch; expected list<number> but got number')
assert_fails('b = N', 'E1012: Type mismatch; expected blob but got number')
assert_fails('Fn = N', 'E1012: Type mismatch; expected func(...): unknown but got number')
assert_fails('A = N', 'E1405: Class "A" cannot be used as a value')
assert_fails('o = N', 'E1012: Type mismatch; expected object<A> but got number')
assert_fails('T = N', 'E1403: Type alias "T" cannot be used as a value')
# Use a compound operator with different LHS types
assert_fails('d += N', 'E734: Wrong variable type for +=')
assert_fails('l += N', 'E734: Wrong variable type for +=')
assert_fails('b += N', 'E734: Wrong variable type for +=')
assert_fails('Fn += N', 'E734: Wrong variable type for +=')
assert_fails('A += N', 'E1405: Class "A" cannot be used as a value')
assert_fails('o += N', 'E734: Wrong variable type for +=')
assert_fails('T += N', 'E1403: Type alias "T" cannot be used as a value')
# Assign to a number variable
assert_fails('N = d', 'E1012: Type mismatch; expected number but got dict<number>')
assert_fails('N = l', 'E1012: Type mismatch; expected number but got list<number>')
assert_fails('N = b', 'E1012: Type mismatch; expected number but got blob')
assert_fails('N = Fn', 'E1012: Type mismatch; expected number but got func([unknown]): number')
assert_fails('N = A', 'E1405: Class "A" cannot be used as a value')
assert_fails('N = o', 'E1012: Type mismatch; expected number but got object<A>')
assert_fails('N = T', 'E1403: Type alias "T" cannot be used as a value')
# Use a compound operator with different RHS types
assert_fails('N += d', 'E734: Wrong variable type for +=')
assert_fails('N += l', 'E734: Wrong variable type for +=')
assert_fails('N += b', 'E974: Using a Blob as a Number')
assert_fails('N += Fn', 'E734: Wrong variable type for +=')
assert_fails('N += A', 'E1405: Class "A" cannot be used as a value')
assert_fails('N += o', 'E1320: Using an Object as a Number')
assert_fails('N += T', 'E1403: Type alias "T" cannot be used as a value')
# Initialize multiple variables using []
assert_fails('var [X1: number, Y: number] = [1, d]', 'E1012: Type mismatch; expected number but got dict<number>')
assert_fails('var [X2: number, Y: number] = [1, l]', 'E1012: Type mismatch; expected number but got list<number>')
assert_fails('var [X3: number, Y: number] = [1, b]', 'E1012: Type mismatch; expected number but got blob')
assert_fails('var [X4: number, Y: number] = [1, Fn]', 'E1012: Type mismatch; expected number but got func([unknown]): number')
assert_fails('var [X7: number, Y: number] = [1, A]', 'E1405: Class "A" cannot be used as a value')
assert_fails('var [X8: number, Y: number] = [1, o]', 'E1012: Type mismatch; expected number but got object<A>')
assert_fails('var [X8: number, Y: number] = [1, T]', 'E1403: Type alias "T" cannot be used as a value')
# String concatenation with various LHS types
assert_fails('S ..= d', 'E734: Wrong variable type for .=')
assert_fails('S ..= l', 'E734: Wrong variable type for .=')
assert_fails('S ..= b', 'E976: Using a Blob as a String')
assert_fails('S ..= Fn', 'E734: Wrong variable type for .=')
assert_fails('S ..= A', 'E1405: Class "A" cannot be used as a value')
assert_fails('S ..= o', 'E1324: Using an Object as a String')
assert_fails('S ..= T', 'E1403: Type alias "T" cannot be used as a value')
# String concatenation with various RHS types
assert_fails('d ..= S', 'E734: Wrong variable type for .=')
assert_fails('l ..= S', 'E734: Wrong variable type for .=')
assert_fails('b ..= S', 'E734: Wrong variable type for .=')
assert_fails('Fn ..= S', 'E734: Wrong variable type for .=')
assert_fails('A ..= S', 'E1405: Class "A" cannot be used as a value')
assert_fails('o ..= S', 'E734: Wrong variable type for .=')
assert_fails('T ..= S', 'E1403: Type alias "T" cannot be used as a value')
END
v9.CheckSourceSuccess(lines)
if has('channel')
lines =<< trim END
vim9script
var N: number = 1
var S: string = 'abc'
var j: job = test_null_job()
var ch: channel = test_null_channel()
assert_fails('j = N', 'E1012: Type mismatch; expected job but got number')
assert_fails('ch = N', 'E1012: Type mismatch; expected channel but got number')
assert_fails('j += N', 'E734: Wrong variable type for +=')
assert_fails('ch += N', 'E734: Wrong variable type for +=')
assert_fails('N = j', 'E1012: Type mismatch; expected number but got job')
assert_fails('N = ch', 'E1012: Type mismatch; expected number but got channel')
assert_fails('N += j', 'E910: Using a Job as a Number')
assert_fails('N += ch', 'E913: Using a Channel as a Number')
assert_fails('var [X5: number, Y: number] = [1, j]', 'E1012: Type mismatch; expected number but got job')
assert_fails('var [X6: number, Y: number] = [1, ch]', 'E1012: Type mismatch; expected number but got channel')
assert_fails('S ..= j', 'E908: Using an invalid value as a String: job')
assert_fails('S ..= ch', 'E908: Using an invalid value as a String: channel')
assert_fails('j ..= S', 'E734: Wrong variable type for .=')
assert_fails('ch ..= S', 'E734: Wrong variable type for .=')
END
v9.CheckSourceSuccess(lines)
endif
lines =<< trim END
vim9script
class A
endclass
def F()
A += 3
enddef
F()
END
v9.CheckScriptFailure(lines, 'E1405: Class "A" cannot be used as a value')
lines =<< trim END
vim9script
class A
endclass
var o = A.new()
def F()
o += 4
enddef
F()
END
v9.CheckScriptFailure(lines, 'E1411: Missing dot after object "o"')
enddef
" Test for checking the argument type of a def function
def Test_func_argtype_check()
var lines =<< trim END
vim9script
# Passing different types as argument to a function expecting a number
def IntArg(n: number)
enddef
class A
endclass
var N: number = 1
var S: string = 'abc'
var d: dict<number> = {}
var l: list<number> = []
var b: blob = 0z10
var Fn: func = function('min')
var o: A = A.new()
assert_fails('IntArg(d)', 'E1013: Argument 1: type mismatch, expected number but got dict<number>')
assert_fails('IntArg(l)', 'E1013: Argument 1: type mismatch, expected number but got list<number>')
assert_fails('IntArg(b)', 'E1013: Argument 1: type mismatch, expected number but got blob')
assert_fails('IntArg(Fn)', 'E1013: Argument 1: type mismatch, expected number but got func([unknown]): number')
if has('channel')
var j: job = test_null_job()
var ch: channel = test_null_channel()
assert_fails('IntArg(j)', 'E1013: Argument 1: type mismatch, expected number but got job')
assert_fails('IntArg(ch)', 'E1013: Argument 1: type mismatch, expected number but got channel')
endif
assert_fails('IntArg(A)', 'E1405: Class "A" cannot be used as a value')
assert_fails('IntArg(o)', 'E1013: Argument 1: type mismatch, expected number but got object<A>')
# Passing a number to functions accepting different argument types
def DictArg(_: dict<number>)
enddef
assert_fails('DictArg(N)', 'E1013: Argument 1: type mismatch, expected dict<number> but got number')
def ListArg(_: list<number>)
enddef
assert_fails('ListArg(N)', 'E1013: Argument 1: type mismatch, expected list<number> but got number')
def BlobArg(_: blob)
enddef
assert_fails('BlobArg(N)', 'E1013: Argument 1: type mismatch, expected blob but got number')
def FuncArg(Fn_arg: func)
enddef
assert_fails('FuncArg(N)', 'E1013: Argument 1: type mismatch, expected func(...): unknown but got number')
if has('channel')
def JobArg(_: job)
enddef
assert_fails('JobArg(N)', 'E1013: Argument 1: type mismatch, expected job but got number')
def ChannelArg(_: channel)
enddef
assert_fails('ChannelArg(N)', 'E1013: Argument 1: type mismatch, expected channel but got number')
endif
def ObjectArg(_: A)
enddef
assert_fails('ObjectArg(N)', 'E1013: Argument 1: type mismatch, expected object<A> but got number')
END
v9.CheckSourceSuccess(lines)
# Calling a function expecting a number type with different argument types
# from another function
var pre_lines =<< trim END
vim9script
class A
endclass
def IntArg(n: number)
enddef
def Foo()
END
var post_lines =<< trim END
enddef
defcompile
END
lines = pre_lines + ['var d: dict<number> = {}', 'IntArg(d)'] + post_lines
v9.CheckSourceFailure(lines, 'E1013: Argument 1: type mismatch, expected number but got dict<number>', 2)
lines = pre_lines + ['var l: list<number> = []', 'IntArg(l)'] + post_lines
v9.CheckSourceFailure(lines, 'E1013: Argument 1: type mismatch, expected number but got list<number>', 2)
lines = pre_lines + ['var b: blob = 0z12', 'IntArg(b)'] + post_lines
v9.CheckSourceFailure(lines, 'E1013: Argument 1: type mismatch, expected number but got blob', 2)
lines = pre_lines + ['var Fn: func = function("min")', 'IntArg(Fn)'] + post_lines
v9.CheckSourceFailure(lines, 'E1013: Argument 1: type mismatch, expected number but got func(...): unknown', 2)
if has('channel')
lines = pre_lines + ['var j: job = test_null_job()', 'IntArg(j)'] + post_lines
v9.CheckSourceFailure(lines, 'E1013: Argument 1: type mismatch, expected number but got job', 2)
lines = pre_lines + ['var ch: channel = test_null_channel()', 'IntArg(ch)'] + post_lines
v9.CheckSourceFailure(lines, 'E1013: Argument 1: type mismatch, expected number but got channel', 2)
endif
lines = pre_lines + ['IntArg(A)'] + post_lines
v9.CheckSourceFailure(lines, 'E1405: Class "A" cannot be used as a value', 1)
lines = pre_lines + ['var o: A = A.new()', 'IntArg(o)'] + post_lines
v9.CheckSourceFailure(lines, 'E1013: Argument 1: type mismatch, expected number but got object<A>', 2)
enddef
" Test for checking the return type of a def function
def Test_func_rettype_check()
var lines =<< trim END
vim9script
def Fn(): dict<number>
return 10
enddef
defcompile
END
v9.CheckSourceFailure(lines, 'E1012: Type mismatch; expected dict<number> but got number', 1)
lines =<< trim END
vim9script
def Fn(): list<number>
return 10
enddef
defcompile
END
v9.CheckSourceFailure(lines, 'E1012: Type mismatch; expected list<number> but got number', 1)
lines =<< trim END
vim9script
def Fn(): blob
return 10
enddef
defcompile
END
v9.CheckSourceFailure(lines, 'E1012: Type mismatch; expected blob but got number', 1)
lines =<< trim END
vim9script
def Fn(): func
return 10
enddef
defcompile
END
v9.CheckSourceFailure(lines, 'E1012: Type mismatch; expected func(...): unknown but got number', 1)
lines =<< trim END
vim9script
def Fn(): job
return 10
enddef
defcompile
END
v9.CheckSourceFailure(lines, 'E1012: Type mismatch; expected job but got number', 1)
lines =<< trim END
vim9script
def Fn(): channel
return 10
enddef
defcompile
END
v9.CheckSourceFailure(lines, 'E1012: Type mismatch; expected channel but got number', 1)
lines =<< trim END
vim9script
class A
endclass
def Fn(): A
return 10
enddef
defcompile
END
v9.CheckSourceFailure(lines, 'E1012: Type mismatch; expected object<A> but got number', 1)
enddef
" Test for assigning different types of value to a variable of type "any"
def Test_assign_to_any()
for [typestr, val] in [
["'bool'", 'true'],
["'number'", '100'],
["'float'", '1.1'],
["'string'", '"abc"'],
["'blob'", '0z10'],
["'list<number>'", '[1, 2, 3]'],
["'dict<number>'", '{a: 1}'],
]
var lines =<< trim eval END
vim9script
var x: any = {val}
assert_equal({typestr}, typename(x))
x = [{{a: 1}}, {{b: 2}}]
assert_equal('list<dict<number>>', typename(x))
def Foo(xarg: any, s: string)
assert_equal(s, typename(xarg))
enddef
Foo({val}, {typestr})
END
v9.CheckSourceSuccess(lines)
endfor
enddef
def Test_assign_type_to_list_dict()
var lines =<< trim END
vim9script
class C
endclass
var x = [C]
END
v9.CheckScriptFailure(lines, 'E1405: Class "C" cannot be used as a value')
lines =<< trim END
vim9script
class C
endclass
type T = C
def F()
var x = [3, T, C]
enddef
F()
END
v9.CheckScriptFailure(lines, 'E1405: Class "C" cannot be used as a value')
lines =<< trim END
vim9script
type T = number
def F()
var x = [3, T]
enddef
F()
END
v9.CheckScriptFailure(lines, 'E1407: Cannot use a Typealias as a variable or value')
lines =<< trim END
vim9script
class C
endclass
var x = {e: C}
END
v9.CheckScriptFailure(lines, 'E1405: Class "C" cannot be used as a value')
lines =<< trim END
vim9script
class C
endclass
def F()
var x = {e: C}
enddef
F()
END
v9.CheckScriptFailure(lines, 'E1405: Class "C" cannot be used as a value')
lines =<< trim END
vim9script
type T = number
def F()
var x = {e: T}
enddef
F()
END
v9.CheckScriptFailure(lines, 'E1407: Cannot use a Typealias as a variable or value')
lines =<< trim END
vim9script
class C
endclass
def F()
var x = {e: [C]}
enddef
F()
END
v9.CheckScriptFailure(lines, 'E1405: Class "C" cannot be used as a value')
lines =<< trim END
vim9script
type T = number
def F()
var x = {e: [T]}
enddef
F()
END
v9.CheckScriptFailure(lines, 'E1407: Cannot use a Typealias as a variable or value')
enddef
" Test for modifying a final variable using a compound operator
def Test_final_var_modification_with_compound_op()
var lines =<< trim END
vim9script
final i: number = 1000
assert_fails('i += 2', 'E46: Cannot change read-only variable "i"')
assert_fails('i -= 2', 'E46: Cannot change read-only variable "i"')
assert_fails('i *= 2', 'E46: Cannot change read-only variable "i"')
assert_fails('i /= 2', 'E46: Cannot change read-only variable "i"')
assert_fails('i %= 2', 'E46: Cannot change read-only variable "i"')
assert_equal(1000, i)
final f: float = 1000.0
assert_fails('f += 2', 'E46: Cannot change read-only variable "f"')
assert_fails('f -= 2', 'E46: Cannot change read-only variable "f"')
assert_fails('f *= 2', 'E46: Cannot change read-only variable "f"')
assert_fails('f /= 2', 'E46: Cannot change read-only variable "f"')
assert_equal(1000.0, f)
final s: string = 'abc'
assert_fails('s ..= "y"', 'E46: Cannot change read-only variable "s"')
assert_equal('abc', s)
END
v9.CheckScriptSuccess(lines)
enddef
" Test for modifying a final variable with a List value
def Test_final_var_with_list_value()
var lines =<< trim END
vim9script
final listA: list<string> = []
var listB = listA
listB->add('a')
assert_true(listA is listB)
assert_equal(['a'], listA)
assert_equal(['a'], listB)
listB += ['b']
assert_true(listA is listB)
assert_equal(['a', 'b'], listA)
assert_equal(['a', 'b'], listB)
listA->add('c')
assert_true(listA is listB)
assert_equal(['a', 'b', 'c'], listA)
assert_equal(['a', 'b', 'c'], listB)
listA += ['d']
assert_true(listA is listB)
assert_equal(['a', 'b', 'c', 'd'], listA)
assert_equal(['a', 'b', 'c', 'd'], listB)
END
v9.CheckScriptSuccess(lines)
enddef
" Test for modifying a final variable with a List value using "+=" from a legacy
" function.
func Test_final_var_with_list_value_legacy()
vim9cmd final g:TestVar = ['a']
vim9cmd g:TestVar += ['b']
call assert_equal(['a', 'b'], g:TestVar)
endfunc
" Test for modifying a final variable with a Blob value
def Test_final_var_with_blob_value()
var lines =<< trim END
vim9script
final blobA: blob = 0z10
var blobB = blobA
blobB->add(32)
assert_true(blobA is blobB)
assert_equal(0z1020, blobA)
assert_equal(0z1020, blobB)
blobB += 0z30
assert_true(blobA is blobB)
assert_equal(0z102030, blobA)
assert_equal(0z102030, blobB)
blobA->add(64)
assert_true(blobA is blobB)
assert_equal(0z10203040, blobA)
assert_equal(0z10203040, blobB)
blobA += 0z50
assert_true(blobA is blobB)
assert_equal(0z1020304050, blobA)
assert_equal(0z1020304050, blobB)
END
v9.CheckScriptSuccess(lines)
enddef
" Test for overwriting a script-local function using the s: dictionary
def Test_override_script_local_func()
var lines =<< trim END
vim9script
def MyFunc()
enddef
var d: dict<any> = s:
d.MyFunc = function('min')
END
v9.CheckScriptFailure(lines, 'E705: Variable name conflicts with existing function: MyFunc', 5)
enddef
" Test for doing a type check at runtime for a list member type
def Test_nested_type_check()
var lines =<< trim END
var d = {a: [10], b: [20]}
var l = d->items()
l[0][1][0] = 'abc'
END
v9.CheckSourceDefExecAndScriptFailure(lines, 'E1012: Type mismatch; expected number but got string')
lines =<< trim END
vim9script
var d = {a: [10], b: [20]}
var l = d->items()
l[0][1][0] = 30
assert_equal([['a', [30]], ['b', [20]]], l)
END
v9.CheckScriptSuccess(lines)
lines =<< trim END
vim9script
def Foo()
var d = {a: [10], b: [20]}
var l = d->items()
l[0][1][0] = 30
assert_equal([['a', [30]], ['b', [20]]], l)
enddef
Foo()
END
v9.CheckScriptSuccess(lines)
# Test for modifying a List item added using add() with a different type.
lines =<< trim END
vim9script
var l: list<list<any>> = [['a']]
var v = [[10]]
l[0]->add(v)
l[0][1][0] = [{x: 20}]
END
v9.CheckScriptFailure(lines, 'E1012: Type mismatch; expected list<number> but got list<dict<number>>', 6)
enddef
" vim: ts=8 sw=2 sts=2 expandtab tw=80 fdm=marker