summaryrefslogtreecommitdiffstats
path: root/src/testdir/test_vim9_script.vim
diff options
context:
space:
mode:
Diffstat (limited to 'src/testdir/test_vim9_script.vim')
-rw-r--r--src/testdir/test_vim9_script.vim3397
1 files changed, 3397 insertions, 0 deletions
diff --git a/src/testdir/test_vim9_script.vim b/src/testdir/test_vim9_script.vim
new file mode 100644
index 0000000..0d9586d
--- /dev/null
+++ b/src/testdir/test_vim9_script.vim
@@ -0,0 +1,3397 @@
+" Test various aspects of the Vim9 script language.
+
+source check.vim
+source term_util.vim
+source view_util.vim
+source vim9.vim
+source shared.vim
+
+def Test_range_only()
+ new
+ setline(1, ['blah', 'Blah'])
+ :/Blah/
+ assert_equal(2, getcurpos()[1])
+ bwipe!
+
+ # without range commands use current line
+ new
+ setline(1, ['one', 'two', 'three'])
+ :2
+ print
+ assert_equal('two', Screenline(&lines))
+ :3
+ list
+ assert_equal('three$', Screenline(&lines))
+ bwipe!
+
+ # won't generate anything
+ if false
+ :123
+ endif
+enddef
+
+let g:alist = [7]
+let g:astring = 'text'
+let g:anumber = 123
+
+def Test_delfunction()
+ # Check function is defined in script namespace
+ CheckScriptSuccess([
+ 'vim9script',
+ 'func CheckMe()',
+ ' return 123',
+ 'endfunc',
+ 'assert_equal(123, s:CheckMe())',
+ ])
+
+ # Check function in script namespace cannot be deleted
+ CheckScriptFailure([
+ 'vim9script',
+ 'func DeleteMe1()',
+ 'endfunc',
+ 'delfunction DeleteMe1',
+ ], 'E1084:')
+ CheckScriptFailure([
+ 'vim9script',
+ 'func DeleteMe2()',
+ 'endfunc',
+ 'def DoThat()',
+ ' delfunction DeleteMe2',
+ 'enddef',
+ 'DoThat()',
+ ], 'E1084:')
+ CheckScriptFailure([
+ 'vim9script',
+ 'def DeleteMe3()',
+ 'enddef',
+ 'delfunction DeleteMe3',
+ ], 'E1084:')
+ CheckScriptFailure([
+ 'vim9script',
+ 'def DeleteMe4()',
+ 'enddef',
+ 'def DoThat()',
+ ' delfunction DeleteMe4',
+ 'enddef',
+ 'DoThat()',
+ ], 'E1084:')
+
+ # Check that global :def function can be replaced and deleted
+ var lines =<< trim END
+ vim9script
+ def g:Global(): string
+ return "yes"
+ enddef
+ assert_equal("yes", g:Global())
+ def! g:Global(): string
+ return "no"
+ enddef
+ assert_equal("no", g:Global())
+ delfunc g:Global
+ assert_false(exists('*g:Global'))
+ END
+ CheckScriptSuccess(lines)
+
+ # Check that global function can be replaced by a :def function and deleted
+ lines =<< trim END
+ vim9script
+ func g:Global()
+ return "yes"
+ endfunc
+ assert_equal("yes", g:Global())
+ def! g:Global(): string
+ return "no"
+ enddef
+ assert_equal("no", g:Global())
+ delfunc g:Global
+ assert_false(exists('*g:Global'))
+ END
+ CheckScriptSuccess(lines)
+
+ # Check that global :def function can be replaced by a function and deleted
+ lines =<< trim END
+ vim9script
+ def g:Global(): string
+ return "yes"
+ enddef
+ assert_equal("yes", g:Global())
+ func! g:Global()
+ return "no"
+ endfunc
+ assert_equal("no", g:Global())
+ delfunc g:Global
+ assert_false(exists('*g:Global'))
+ END
+ CheckScriptSuccess(lines)
+enddef
+
+def Test_wrong_type()
+ CheckDefFailure(['var name: list<nothing>'], 'E1010:')
+ CheckDefFailure(['var name: list<list<nothing>>'], 'E1010:')
+ CheckDefFailure(['var name: dict<nothing>'], 'E1010:')
+ CheckDefFailure(['var name: dict<dict<nothing>>'], 'E1010:')
+
+ CheckDefFailure(['var name: dict<number'], 'E1009:')
+ CheckDefFailure(['var name: dict<list<number>'], 'E1009:')
+
+ CheckDefFailure(['var name: ally'], 'E1010:')
+ CheckDefFailure(['var name: bram'], 'E1010:')
+ CheckDefFailure(['var name: cathy'], 'E1010:')
+ CheckDefFailure(['var name: dom'], 'E1010:')
+ CheckDefFailure(['var name: freddy'], 'E1010:')
+ CheckDefFailure(['var name: john'], 'E1010:')
+ CheckDefFailure(['var name: larry'], 'E1010:')
+ CheckDefFailure(['var name: ned'], 'E1010:')
+ CheckDefFailure(['var name: pam'], 'E1010:')
+ CheckDefFailure(['var name: sam'], 'E1010:')
+ CheckDefFailure(['var name: vim'], 'E1010:')
+
+ CheckDefFailure(['var Ref: number', 'Ref()'], 'E1085:')
+ CheckDefFailure(['var Ref: string', 'var res = Ref()'], 'E1085:')
+enddef
+
+def Test_script_wrong_type()
+ var lines =<< trim END
+ vim9script
+ var s:dict: dict<string>
+ s:dict['a'] = ['x']
+ END
+ CheckScriptFailure(lines, 'E1012: Type mismatch; expected string but got list<string>', 3)
+enddef
+
+def Test_const()
+ CheckDefFailure(['final name = 234', 'name = 99'], 'E1018:')
+ CheckDefFailure(['final one = 234', 'var one = 99'], 'E1017:')
+ CheckDefFailure(['final list = [1, 2]', 'var list = [3, 4]'], 'E1017:')
+ CheckDefFailure(['final two'], 'E1125:')
+ CheckDefFailure(['final &option'], 'E996:')
+
+ var lines =<< trim END
+ final list = [1, 2, 3]
+ list[0] = 4
+ list->assert_equal([4, 2, 3])
+ const other = [5, 6, 7]
+ other->assert_equal([5, 6, 7])
+
+ var varlist = [7, 8]
+ const constlist = [1, varlist, 3]
+ varlist[0] = 77
+ # TODO: does not work yet
+ # constlist[1][1] = 88
+ var cl = constlist[1]
+ cl[1] = 88
+ constlist->assert_equal([1, [77, 88], 3])
+
+ var vardict = {five: 5, six: 6}
+ const constdict = {one: 1, two: vardict, three: 3}
+ vardict['five'] = 55
+ # TODO: does not work yet
+ # constdict['two']['six'] = 66
+ var cd = constdict['two']
+ cd['six'] = 66
+ constdict->assert_equal({one: 1, two: {five: 55, six: 66}, three: 3})
+ END
+ CheckDefAndScriptSuccess(lines)
+enddef
+
+def Test_const_bang()
+ var lines =<< trim END
+ const var = 234
+ var = 99
+ END
+ CheckDefExecFailure(lines, 'E1018:', 2)
+ CheckScriptFailure(['vim9script'] + lines, 'E46:', 3)
+
+ lines =<< trim END
+ const ll = [2, 3, 4]
+ ll[0] = 99
+ END
+ CheckDefExecFailure(lines, 'E1119:', 2)
+ CheckScriptFailure(['vim9script'] + lines, 'E741:', 3)
+
+ lines =<< trim END
+ const ll = [2, 3, 4]
+ ll[3] = 99
+ END
+ CheckDefExecFailure(lines, 'E1118:', 2)
+ CheckScriptFailure(['vim9script'] + lines, 'E684:', 3)
+
+ lines =<< trim END
+ const dd = {one: 1, two: 2}
+ dd["one"] = 99
+ END
+ CheckDefExecFailure(lines, 'E1121:', 2)
+ CheckScriptFailure(['vim9script'] + lines, 'E741:', 3)
+
+ lines =<< trim END
+ const dd = {one: 1, two: 2}
+ dd["three"] = 99
+ END
+ CheckDefExecFailure(lines, 'E1120:')
+ CheckScriptFailure(['vim9script'] + lines, 'E741:', 3)
+enddef
+
+def Test_range_no_colon()
+ CheckDefFailure(['%s/a/b/'], 'E1050:')
+ CheckDefFailure(['+ s/a/b/'], 'E1050:')
+ CheckDefFailure(['- s/a/b/'], 'E1050:')
+ CheckDefFailure(['. s/a/b/'], 'E1050:')
+enddef
+
+
+def Test_block()
+ var outer = 1
+ {
+ var inner = 2
+ assert_equal(1, outer)
+ assert_equal(2, inner)
+ }
+ assert_equal(1, outer)
+enddef
+
+def Test_block_failure()
+ CheckDefFailure(['{', 'var inner = 1', '}', 'echo inner'], 'E1001:')
+ CheckDefFailure(['}'], 'E1025:')
+ CheckDefFailure(['{', 'echo 1'], 'E1026:')
+enddef
+
+def Test_block_local_vars()
+ var lines =<< trim END
+ vim9script
+ v:testing = 1
+ if true
+ var text = ['hello']
+ def SayHello(): list<string>
+ return text
+ enddef
+ def SetText(v: string)
+ text = [v]
+ enddef
+ endif
+
+ if true
+ var text = ['again']
+ def SayAgain(): list<string>
+ return text
+ enddef
+ endif
+
+ # test that the "text" variables are not cleaned up
+ test_garbagecollect_now()
+
+ defcompile
+
+ assert_equal(['hello'], SayHello())
+ assert_equal(['again'], SayAgain())
+
+ SetText('foobar')
+ assert_equal(['foobar'], SayHello())
+
+ call writefile(['ok'], 'Xdidit')
+ qall!
+ END
+
+ # need to execute this with a separate Vim instance to avoid the current
+ # context gets garbage collected.
+ writefile(lines, 'Xscript')
+ RunVim([], [], '-S Xscript')
+ assert_equal(['ok'], readfile('Xdidit'))
+
+ delete('Xscript')
+ delete('Xdidit')
+enddef
+
+def Test_block_local_vars_with_func()
+ var lines =<< trim END
+ vim9script
+ if true
+ var foo = 'foo'
+ if true
+ var bar = 'bar'
+ def Func(): list<string>
+ return [foo, bar]
+ enddef
+ endif
+ endif
+ # function is compiled here, after blocks have finished, can still access
+ # "foo" and "bar"
+ assert_equal(['foo', 'bar'], Func())
+ END
+ CheckScriptSuccess(lines)
+enddef
+
+func g:NoSuchFunc()
+ echo 'none'
+endfunc
+
+def Test_try_catch_throw()
+ var l = []
+ try # comment
+ add(l, '1')
+ throw 'wrong'
+ add(l, '2')
+ catch # comment
+ add(l, v:exception)
+ finally # comment
+ add(l, '3')
+ endtry # comment
+ assert_equal(['1', 'wrong', '3'], l)
+
+ l = []
+ try
+ try
+ add(l, '1')
+ throw 'wrong'
+ add(l, '2')
+ catch /right/
+ add(l, v:exception)
+ endtry
+ catch /wrong/
+ add(l, 'caught')
+ fina
+ add(l, 'finally')
+ endtry
+ assert_equal(['1', 'caught', 'finally'], l)
+
+ var n: number
+ try
+ n = l[3]
+ catch /E684:/
+ n = 99
+ endtry
+ assert_equal(99, n)
+
+ var done = 'no'
+ if 0
+ try | catch | endtry
+ else
+ done = 'yes'
+ endif
+ assert_equal('yes', done)
+
+ done = 'no'
+ if 1
+ done = 'yes'
+ else
+ try | catch | endtry
+ done = 'never'
+ endif
+ assert_equal('yes', done)
+
+ if 1
+ else
+ try | catch /pat/ | endtry
+ try | catch /pat/
+ endtry
+ try
+ catch /pat/ | endtry
+ try
+ catch /pat/
+ endtry
+ endif
+
+ try
+ # string slice returns a string, not a number
+ n = g:astring[3]
+ catch /E1012:/
+ n = 77
+ endtry
+ assert_equal(77, n)
+
+ try
+ n = l[g:astring]
+ catch /E1012:/
+ n = 88
+ endtry
+ assert_equal(88, n)
+
+ try
+ n = s:does_not_exist
+ catch /E121:/
+ n = 111
+ endtry
+ assert_equal(111, n)
+
+ try
+ n = g:does_not_exist
+ catch /E121:/
+ n = 121
+ endtry
+ assert_equal(121, n)
+
+ var d = {one: 1}
+ try
+ n = d[g:astring]
+ catch /E716:/
+ n = 222
+ endtry
+ assert_equal(222, n)
+
+ try
+ n = -g:astring
+ catch /E39:/
+ n = 233
+ endtry
+ assert_equal(233, n)
+
+ try
+ n = +g:astring
+ catch /E1030:/
+ n = 244
+ endtry
+ assert_equal(244, n)
+
+ try
+ n = +g:alist
+ catch /E745:/
+ n = 255
+ endtry
+ assert_equal(255, n)
+
+ var nd: dict<any>
+ try
+ nd = {[g:anumber]: 1}
+ catch /E1012:/
+ n = 266
+ endtry
+ assert_equal(266, n)
+
+ try
+ [n] = [1, 2, 3]
+ catch /E1093:/
+ n = 277
+ endtry
+ assert_equal(277, n)
+
+ try
+ &ts = g:astring
+ catch /E1012:/
+ n = 288
+ endtry
+ assert_equal(288, n)
+
+ try
+ &backspace = 'asdf'
+ catch /E474:/
+ n = 299
+ endtry
+ assert_equal(299, n)
+
+ l = [1]
+ try
+ l[3] = 3
+ catch /E684:/
+ n = 300
+ endtry
+ assert_equal(300, n)
+
+ try
+ unlet g:does_not_exist
+ catch /E108:/
+ n = 322
+ endtry
+ assert_equal(322, n)
+
+ try
+ d = {text: 1, [g:astring]: 2}
+ catch /E721:/
+ n = 333
+ endtry
+ assert_equal(333, n)
+
+ try
+ l = DeletedFunc()
+ catch /E933:/
+ n = 344
+ endtry
+ assert_equal(344, n)
+
+ try
+ echo len(v:true)
+ catch /E701:/
+ n = 355
+ endtry
+ assert_equal(355, n)
+
+ var P = function('g:NoSuchFunc')
+ delfunc g:NoSuchFunc
+ try
+ echo P()
+ catch /E117:/
+ n = 366
+ endtry
+ assert_equal(366, n)
+
+ try
+ echo g:NoSuchFunc()
+ catch /E117:/
+ n = 377
+ endtry
+ assert_equal(377, n)
+
+ try
+ echo g:alist + 4
+ catch /E745:/
+ n = 388
+ endtry
+ assert_equal(388, n)
+
+ try
+ echo 4 + g:alist
+ catch /E745:/
+ n = 399
+ endtry
+ assert_equal(399, n)
+
+ try
+ echo g:alist.member
+ catch /E715:/
+ n = 400
+ endtry
+ assert_equal(400, n)
+
+ try
+ echo d.member
+ catch /E716:/
+ n = 411
+ endtry
+ assert_equal(411, n)
+enddef
+
+def Test_cnext_works_in_catch()
+ var lines =<< trim END
+ vim9script
+ au BufEnter * eval 0
+ writefile(['text'], 'Xfile1')
+ writefile(['text'], 'Xfile2')
+ var items = [
+ {lnum: 1, filename: 'Xfile1', valid: true},
+ {lnum: 1, filename: 'Xfile2', valid: true}
+ ]
+ setqflist([], ' ', {items: items})
+ cwindow
+
+ def CnextOrCfirst()
+ # if cnext fails, cfirst is used
+ try
+ cnext
+ catch
+ cfirst
+ endtry
+ enddef
+
+ CnextOrCfirst()
+ CnextOrCfirst()
+ writefile([getqflist({idx: 0}).idx], 'Xresult')
+ qall
+ END
+ writefile(lines, 'XCatchCnext')
+ RunVim([], [], '--clean -S XCatchCnext')
+ assert_equal(['1'], readfile('Xresult'))
+
+ delete('Xfile1')
+ delete('Xfile2')
+ delete('XCatchCnext')
+ delete('Xresult')
+enddef
+
+def Test_throw_skipped()
+ if 0
+ throw dontgethere
+ endif
+enddef
+
+def Test_nocatch_throw_silenced()
+ var lines =<< trim END
+ vim9script
+ def Func()
+ throw 'error'
+ enddef
+ silent! Func()
+ END
+ writefile(lines, 'XthrowSilenced')
+ source XthrowSilenced
+ delete('XthrowSilenced')
+enddef
+
+def DeletedFunc(): list<any>
+ return ['delete me']
+enddef
+defcompile
+delfunc DeletedFunc
+
+def ThrowFromDef()
+ throw "getout" # comment
+enddef
+
+func CatchInFunc()
+ try
+ call ThrowFromDef()
+ catch
+ let g:thrown_func = v:exception
+ endtry
+endfunc
+
+def CatchInDef()
+ try
+ ThrowFromDef()
+ catch
+ g:thrown_def = v:exception
+ endtry
+enddef
+
+def ReturnFinally(): string
+ try
+ return 'intry'
+ finall
+ g:in_finally = 'finally'
+ endtry
+ return 'end'
+enddef
+
+def Test_try_catch_nested()
+ CatchInFunc()
+ assert_equal('getout', g:thrown_func)
+
+ CatchInDef()
+ assert_equal('getout', g:thrown_def)
+
+ assert_equal('intry', ReturnFinally())
+ assert_equal('finally', g:in_finally)
+enddef
+
+def TryOne(): number
+ try
+ return 0
+ catch
+ endtry
+ return 0
+enddef
+
+def TryTwo(n: number): string
+ try
+ var x = {}
+ catch
+ endtry
+ return 'text'
+enddef
+
+def Test_try_catch_twice()
+ assert_equal('text', TryOne()->TryTwo())
+enddef
+
+def Test_try_catch_match()
+ var seq = 'a'
+ try
+ throw 'something'
+ catch /nothing/
+ seq ..= 'x'
+ catch /some/
+ seq ..= 'b'
+ catch /asdf/
+ seq ..= 'x'
+ catch ?a\?sdf?
+ seq ..= 'y'
+ finally
+ seq ..= 'c'
+ endtry
+ assert_equal('abc', seq)
+enddef
+
+def Test_try_catch_fails()
+ CheckDefFailure(['catch'], 'E603:')
+ CheckDefFailure(['try', 'echo 0', 'catch', 'catch'], 'E1033:')
+ CheckDefFailure(['try', 'echo 0', 'catch /pat'], 'E1067:')
+ CheckDefFailure(['finally'], 'E606:')
+ CheckDefFailure(['try', 'echo 0', 'finally', 'echo 1', 'finally'], 'E607:')
+ CheckDefFailure(['endtry'], 'E602:')
+ CheckDefFailure(['while 1', 'endtry'], 'E170:')
+ CheckDefFailure(['for i in range(5)', 'endtry'], 'E170:')
+ CheckDefFailure(['if 1', 'endtry'], 'E171:')
+ CheckDefFailure(['try', 'echo 1', 'endtry'], 'E1032:')
+
+ CheckDefFailure(['throw'], 'E1143:')
+ CheckDefFailure(['throw xxx'], 'E1001:')
+enddef
+
+def Test_throw_vimscript()
+ # only checks line continuation
+ var lines =<< trim END
+ vim9script
+ try
+ throw 'one'
+ .. 'two'
+ catch
+ assert_equal('onetwo', v:exception)
+ endtry
+ END
+ CheckScriptSuccess(lines)
+
+ lines =<< trim END
+ vim9script
+ @r = ''
+ def Func()
+ throw @r
+ enddef
+ var result = ''
+ try
+ Func()
+ catch /E1129:/
+ result = 'caught'
+ endtry
+ assert_equal('caught', result)
+ END
+ CheckScriptSuccess(lines)
+enddef
+
+def Test_error_in_nested_function()
+ # an error in a nested :function aborts executin in the calling :def function
+ var lines =<< trim END
+ vim9script
+ def Func()
+ Error()
+ g:test_var = 1
+ enddef
+ func Error() abort
+ eval [][0]
+ endfunc
+ Func()
+ END
+ g:test_var = 0
+ CheckScriptFailure(lines, 'E684:')
+ assert_equal(0, g:test_var)
+enddef
+
+def Test_cexpr_vimscript()
+ # only checks line continuation
+ set errorformat=File\ %f\ line\ %l
+ var lines =<< trim END
+ vim9script
+ cexpr 'File'
+ .. ' someFile' ..
+ ' line 19'
+ assert_equal(19, getqflist()[0].lnum)
+ END
+ CheckScriptSuccess(lines)
+ set errorformat&
+enddef
+
+def Test_statusline_syntax()
+ # legacy syntax is used for 'statusline'
+ var lines =<< trim END
+ vim9script
+ func g:Status()
+ return '%{"x" is# "x"}'
+ endfunc
+ set laststatus=2 statusline=%!Status()
+ redrawstatus
+ set laststatus statusline=
+ END
+ CheckScriptSuccess(lines)
+enddef
+
+def Test_list_vimscript()
+ # checks line continuation and comments
+ var lines =<< trim END
+ vim9script
+ var mylist = [
+ 'one',
+ # comment
+ 'two', # empty line follows
+
+ 'three',
+ ]
+ assert_equal(['one', 'two', 'three'], mylist)
+ END
+ CheckScriptSuccess(lines)
+
+ # check all lines from heredoc are kept
+ lines =<< trim END
+ # comment 1
+ two
+ # comment 3
+
+ five
+ # comment 6
+ END
+ assert_equal(['# comment 1', 'two', '# comment 3', '', 'five', '# comment 6'], lines)
+
+ lines =<< trim END
+ [{
+ a: 0}]->string()->assert_equal("[{'a': 0}]")
+ END
+ CheckDefAndScriptSuccess(lines)
+enddef
+
+if has('channel')
+ let someJob = test_null_job()
+
+ def FuncWithError()
+ echomsg g:someJob
+ enddef
+
+ func Test_convert_emsg_to_exception()
+ try
+ call FuncWithError()
+ catch
+ call assert_match('Vim:E908:', v:exception)
+ endtry
+ endfunc
+endif
+
+let s:export_script_lines =<< trim END
+ vim9script
+ var name: string = 'bob'
+ def Concat(arg: string): string
+ return name .. arg
+ enddef
+ g:result = Concat('bie')
+ g:localname = name
+
+ export const CONST = 1234
+ export var exported = 9876
+ export var exp_name = 'John'
+ export def Exported(): string
+ return 'Exported'
+ enddef
+END
+
+def Undo_export_script_lines()
+ unlet g:result
+ unlet g:localname
+enddef
+
+def Test_vim9_import_export()
+ var import_script_lines =<< trim END
+ vim9script
+ import {exported, Exported} from './Xexport.vim'
+ g:imported = exported
+ exported += 3
+ g:imported_added = exported
+ g:imported_func = Exported()
+
+ def GetExported(): string
+ var local_dict = {ref: Exported}
+ return local_dict.ref()
+ enddef
+ g:funcref_result = GetExported()
+
+ import {exp_name} from './Xexport.vim'
+ g:imported_name = exp_name
+ exp_name ..= ' Doe'
+ g:imported_name_appended = exp_name
+ g:imported_later = exported
+ END
+
+ writefile(import_script_lines, 'Ximport.vim')
+ writefile(s:export_script_lines, 'Xexport.vim')
+
+ source Ximport.vim
+
+ assert_equal('bobbie', g:result)
+ assert_equal('bob', g:localname)
+ assert_equal(9876, g:imported)
+ assert_equal(9879, g:imported_added)
+ assert_equal(9879, g:imported_later)
+ assert_equal('Exported', g:imported_func)
+ assert_equal('Exported', g:funcref_result)
+ assert_equal('John', g:imported_name)
+ assert_equal('John Doe', g:imported_name_appended)
+ assert_false(exists('g:name'))
+
+ Undo_export_script_lines()
+ unlet g:imported
+ unlet g:imported_added
+ unlet g:imported_later
+ unlet g:imported_func
+ unlet g:imported_name g:imported_name_appended
+ delete('Ximport.vim')
+
+ # similar, with line breaks
+ var import_line_break_script_lines =<< trim END
+ vim9script
+ import {
+ exported,
+ Exported,
+ }
+ from
+ './Xexport.vim'
+ g:imported = exported
+ exported += 5
+ g:imported_added = exported
+ g:imported_func = Exported()
+ END
+ writefile(import_line_break_script_lines, 'Ximport_lbr.vim')
+ source Ximport_lbr.vim
+
+ assert_equal(9876, g:imported)
+ assert_equal(9881, g:imported_added)
+ assert_equal('Exported', g:imported_func)
+
+ # exported script not sourced again
+ assert_false(exists('g:result'))
+ unlet g:imported
+ unlet g:imported_added
+ unlet g:imported_func
+ delete('Ximport_lbr.vim')
+
+ # import inside :def function
+ var import_in_def_lines =<< trim END
+ vim9script
+ def ImportInDef()
+ import exported from './Xexport.vim'
+ g:imported = exported
+ exported += 7
+ g:imported_added = exported
+ enddef
+ ImportInDef()
+ END
+ writefile(import_in_def_lines, 'Ximport2.vim')
+ source Ximport2.vim
+ # TODO: this should be 9879
+ assert_equal(9876, g:imported)
+ assert_equal(9883, g:imported_added)
+ unlet g:imported
+ unlet g:imported_added
+ delete('Ximport2.vim')
+
+ var import_star_as_lines =<< trim END
+ vim9script
+ import * as Export from './Xexport.vim'
+ def UseExport()
+ g:imported = Export.exported
+ enddef
+ UseExport()
+ END
+ writefile(import_star_as_lines, 'Ximport.vim')
+ source Ximport.vim
+ assert_equal(9883, g:imported)
+
+ var import_star_as_lines_no_dot =<< trim END
+ vim9script
+ import * as Export from './Xexport.vim'
+ def Func()
+ var dummy = 1
+ var imported = Export + dummy
+ enddef
+ defcompile
+ END
+ writefile(import_star_as_lines_no_dot, 'Ximport.vim')
+ assert_fails('source Ximport.vim', 'E1060:', '', 2, 'Func')
+
+ var import_star_as_lines_dot_space =<< trim END
+ vim9script
+ import * as Export from './Xexport.vim'
+ def Func()
+ var imported = Export . exported
+ enddef
+ defcompile
+ END
+ writefile(import_star_as_lines_dot_space, 'Ximport.vim')
+ assert_fails('source Ximport.vim', 'E1074:', '', 1, 'Func')
+
+ var import_star_as_duplicated =<< trim END
+ vim9script
+ import * as Export from './Xexport.vim'
+ var some = 'other'
+ import * as Export from './Xexport.vim'
+ defcompile
+ END
+ writefile(import_star_as_duplicated, 'Ximport.vim')
+ assert_fails('source Ximport.vim', 'E1073:', '', 4, 'Ximport.vim')
+
+ var import_star_as_lines_missing_name =<< trim END
+ vim9script
+ import * as Export from './Xexport.vim'
+ def Func()
+ var imported = Export.
+ enddef
+ defcompile
+ END
+ writefile(import_star_as_lines_missing_name, 'Ximport.vim')
+ assert_fails('source Ximport.vim', 'E1048:', '', 1, 'Func')
+
+ var import_star_as_lbr_lines =<< trim END
+ vim9script
+ import *
+ as Export
+ from
+ './Xexport.vim'
+ def UseExport()
+ g:imported = Export.exported
+ enddef
+ UseExport()
+ END
+ writefile(import_star_as_lbr_lines, 'Ximport.vim')
+ source Ximport.vim
+ assert_equal(9883, g:imported)
+
+ var import_star_lines =<< trim END
+ vim9script
+ import * from './Xexport.vim'
+ END
+ writefile(import_star_lines, 'Ximport.vim')
+ assert_fails('source Ximport.vim', 'E1045:', '', 2, 'Ximport.vim')
+
+ # try to import something that exists but is not exported
+ var import_not_exported_lines =<< trim END
+ vim9script
+ import name from './Xexport.vim'
+ END
+ writefile(import_not_exported_lines, 'Ximport.vim')
+ assert_fails('source Ximport.vim', 'E1049:', '', 2, 'Ximport.vim')
+
+ # try to import something that is already defined
+ var import_already_defined =<< trim END
+ vim9script
+ var exported = 'something'
+ import exported from './Xexport.vim'
+ END
+ writefile(import_already_defined, 'Ximport.vim')
+ assert_fails('source Ximport.vim', 'E1073:', '', 3, 'Ximport.vim')
+
+ # try to import something that is already defined
+ import_already_defined =<< trim END
+ vim9script
+ var exported = 'something'
+ import * as exported from './Xexport.vim'
+ END
+ writefile(import_already_defined, 'Ximport.vim')
+ assert_fails('source Ximport.vim', 'E1073:', '', 3, 'Ximport.vim')
+
+ # try to import something that is already defined
+ import_already_defined =<< trim END
+ vim9script
+ var exported = 'something'
+ import {exported} from './Xexport.vim'
+ END
+ writefile(import_already_defined, 'Ximport.vim')
+ assert_fails('source Ximport.vim', 'E1073:', '', 3, 'Ximport.vim')
+
+ # try changing an imported const
+ var import_assign_to_const =<< trim END
+ vim9script
+ import CONST from './Xexport.vim'
+ def Assign()
+ CONST = 987
+ enddef
+ defcompile
+ END
+ writefile(import_assign_to_const, 'Ximport.vim')
+ assert_fails('source Ximport.vim', 'E46:', '', 1, '_Assign')
+
+ # import a very long name, requires making a copy
+ var import_long_name_lines =<< trim END
+ vim9script
+ import name012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789 from './Xexport.vim'
+ END
+ writefile(import_long_name_lines, 'Ximport.vim')
+ assert_fails('source Ximport.vim', 'E1048:', '', 2, 'Ximport.vim')
+
+ var import_no_from_lines =<< trim END
+ vim9script
+ import name './Xexport.vim'
+ END
+ writefile(import_no_from_lines, 'Ximport.vim')
+ assert_fails('source Ximport.vim', 'E1070:', '', 2, 'Ximport.vim')
+
+ var import_invalid_string_lines =<< trim END
+ vim9script
+ import name from Xexport.vim
+ END
+ writefile(import_invalid_string_lines, 'Ximport.vim')
+ assert_fails('source Ximport.vim', 'E1071:', '', 2, 'Ximport.vim')
+
+ var import_wrong_name_lines =<< trim END
+ vim9script
+ import name from './XnoExport.vim'
+ END
+ writefile(import_wrong_name_lines, 'Ximport.vim')
+ assert_fails('source Ximport.vim', 'E1053:', '', 2, 'Ximport.vim')
+
+ var import_missing_comma_lines =<< trim END
+ vim9script
+ import {exported name} from './Xexport.vim'
+ END
+ writefile(import_missing_comma_lines, 'Ximport3.vim')
+ assert_fails('source Ximport3.vim', 'E1046:', '', 2, 'Ximport3.vim')
+
+ delete('Ximport.vim')
+ delete('Ximport3.vim')
+ delete('Xexport.vim')
+
+ # Check that in a Vim9 script 'cpo' is set to the Vim default.
+ set cpo&vi
+ var cpo_before = &cpo
+ var lines =<< trim END
+ vim9script
+ g:cpo_in_vim9script = &cpo
+ END
+ writefile(lines, 'Xvim9_script')
+ source Xvim9_script
+ assert_equal(cpo_before, &cpo)
+ set cpo&vim
+ assert_equal(&cpo, g:cpo_in_vim9script)
+ delete('Xvim9_script')
+enddef
+
+func g:Trigger()
+ source Ximport.vim
+ return "echo 'yes'\<CR>"
+endfunc
+
+def Test_import_export_expr_map()
+ # check that :import and :export work when buffer is locked
+ var export_lines =<< trim END
+ vim9script
+ export def That(): string
+ return 'yes'
+ enddef
+ END
+ writefile(export_lines, 'Xexport_that.vim')
+
+ var import_lines =<< trim END
+ vim9script
+ import That from './Xexport_that.vim'
+ assert_equal('yes', That())
+ END
+ writefile(import_lines, 'Ximport.vim')
+
+ nnoremap <expr> trigger g:Trigger()
+ feedkeys('trigger', "xt")
+
+ delete('Xexport_that.vim')
+ delete('Ximport.vim')
+ nunmap trigger
+enddef
+
+def Test_import_in_filetype()
+ # check that :import works when the buffer is locked
+ mkdir('ftplugin', 'p')
+ var export_lines =<< trim END
+ vim9script
+ export var That = 'yes'
+ END
+ writefile(export_lines, 'ftplugin/Xexport_ft.vim')
+
+ var import_lines =<< trim END
+ vim9script
+ import That from './Xexport_ft.vim'
+ assert_equal('yes', That)
+ g:did_load_mytpe = 1
+ END
+ writefile(import_lines, 'ftplugin/qf.vim')
+
+ var save_rtp = &rtp
+ &rtp = getcwd() .. ',' .. &rtp
+
+ filetype plugin on
+ copen
+ assert_equal(1, g:did_load_mytpe)
+
+ quit!
+ delete('Xexport_ft.vim')
+ delete('ftplugin', 'rf')
+ &rtp = save_rtp
+enddef
+
+def Test_use_import_in_mapping()
+ var lines =<< trim END
+ vim9script
+ export def Funcx()
+ g:result = 42
+ enddef
+ END
+ writefile(lines, 'XsomeExport.vim')
+ lines =<< trim END
+ vim9script
+ import Funcx from './XsomeExport.vim'
+ nnoremap <F3> :call <sid>Funcx()<cr>
+ END
+ writefile(lines, 'Xmapscript.vim')
+
+ source Xmapscript.vim
+ feedkeys("\<F3>", "xt")
+ assert_equal(42, g:result)
+
+ unlet g:result
+ delete('XsomeExport.vim')
+ delete('Xmapscript.vim')
+ nunmap <F3>
+enddef
+
+def Test_vim9script_fails()
+ CheckScriptFailure(['scriptversion 2', 'vim9script'], 'E1039:')
+ CheckScriptFailure(['vim9script', 'scriptversion 2'], 'E1040:')
+ CheckScriptFailure(['export var some = 123'], 'E1042:')
+ CheckScriptFailure(['import some from "./Xexport.vim"'], 'E1048:')
+ CheckScriptFailure(['vim9script', 'export var g:some'], 'E1022:')
+ CheckScriptFailure(['vim9script', 'export echo 134'], 'E1043:')
+
+ CheckScriptFailure(['vim9script', 'var str: string', 'str = 1234'], 'E1012:')
+ CheckScriptFailure(['vim9script', 'const str = "asdf"', 'str = "xxx"'], 'E46:')
+
+ assert_fails('vim9script', 'E1038:')
+ assert_fails('export something', 'E1043:')
+enddef
+
+func Test_import_fails_without_script()
+ CheckRunVimInTerminal
+
+ " call indirectly to avoid compilation error for missing functions
+ call Run_Test_import_fails_on_command_line()
+endfunc
+
+def Run_Test_import_fails_on_command_line()
+ var export =<< trim END
+ vim9script
+ export def Foo(): number
+ return 0
+ enddef
+ END
+ writefile(export, 'XexportCmd.vim')
+
+ var buf = RunVimInTerminal('-c "import Foo from ''./XexportCmd.vim''"', {
+ rows: 6, wait_for_ruler: 0})
+ WaitForAssert(() => assert_match('^E1094:', term_getline(buf, 5)))
+
+ delete('XexportCmd.vim')
+ StopVimInTerminal(buf)
+enddef
+
+def Test_vim9script_reload_noclear()
+ var lines =<< trim END
+ vim9script
+ export var exported = 'thexport'
+ END
+ writefile(lines, 'XExportReload')
+ lines =<< trim END
+ vim9script noclear
+ g:loadCount += 1
+ var s:reloaded = 'init'
+ import exported from './XExportReload'
+
+ def Again(): string
+ return 'again'
+ enddef
+
+ if exists('s:loaded') | finish | endif
+ var s:loaded = true
+
+ var s:notReloaded = 'yes'
+ s:reloaded = 'first'
+ def g:Values(): list<string>
+ return [s:reloaded, s:notReloaded, Again(), Once(), exported]
+ enddef
+
+ def Once(): string
+ return 'once'
+ enddef
+ END
+ writefile(lines, 'XReloaded')
+ g:loadCount = 0
+ source XReloaded
+ assert_equal(1, g:loadCount)
+ assert_equal(['first', 'yes', 'again', 'once', 'thexport'], g:Values())
+ source XReloaded
+ assert_equal(2, g:loadCount)
+ assert_equal(['init', 'yes', 'again', 'once', 'thexport'], g:Values())
+ source XReloaded
+ assert_equal(3, g:loadCount)
+ assert_equal(['init', 'yes', 'again', 'once', 'thexport'], g:Values())
+
+ delete('XReloaded')
+ delete('XExportReload')
+ delfunc g:Values
+ unlet g:loadCount
+enddef
+
+def Test_vim9script_reload_import()
+ var lines =<< trim END
+ vim9script
+ const var = ''
+ var valone = 1234
+ def MyFunc(arg: string)
+ valone = 5678
+ enddef
+ END
+ var morelines =<< trim END
+ var valtwo = 222
+ export def GetValtwo(): number
+ return valtwo
+ enddef
+ END
+ writefile(lines + morelines, 'Xreload.vim')
+ source Xreload.vim
+ source Xreload.vim
+ source Xreload.vim
+
+ var testlines =<< trim END
+ vim9script
+ def TheFunc()
+ import GetValtwo from './Xreload.vim'
+ assert_equal(222, GetValtwo())
+ enddef
+ TheFunc()
+ END
+ writefile(testlines, 'Ximport.vim')
+ source Ximport.vim
+
+ # Test that when not using "morelines" GetValtwo() and valtwo are still
+ # defined, because import doesn't reload a script.
+ writefile(lines, 'Xreload.vim')
+ source Ximport.vim
+
+ # cannot declare a var twice
+ lines =<< trim END
+ vim9script
+ var valone = 1234
+ var valone = 5678
+ END
+ writefile(lines, 'Xreload.vim')
+ assert_fails('source Xreload.vim', 'E1041:', '', 3, 'Xreload.vim')
+
+ delete('Xreload.vim')
+ delete('Ximport.vim')
+enddef
+
+" if a script is reloaded with a script-local variable that changed its type, a
+" compiled function using that variable must fail.
+def Test_script_reload_change_type()
+ var lines =<< trim END
+ vim9script noclear
+ var str = 'string'
+ def g:GetStr(): string
+ return str .. 'xxx'
+ enddef
+ END
+ writefile(lines, 'Xreload.vim')
+ source Xreload.vim
+ echo g:GetStr()
+
+ lines =<< trim END
+ vim9script noclear
+ var str = 1234
+ END
+ writefile(lines, 'Xreload.vim')
+ source Xreload.vim
+ assert_fails('echo g:GetStr()', 'E1150:')
+
+ delfunc g:GetStr
+ delete('Xreload.vim')
+enddef
+
+def s:RetSome(): string
+ return 'some'
+enddef
+
+" Not exported function that is referenced needs to be accessed by the
+" script-local name.
+def Test_vim9script_funcref()
+ var sortlines =<< trim END
+ vim9script
+ def Compare(i1: number, i2: number): number
+ return i2 - i1
+ enddef
+
+ export def FastSort(): list<number>
+ return range(5)->sort(Compare)
+ enddef
+ END
+ writefile(sortlines, 'Xsort.vim')
+
+ var lines =<< trim END
+ vim9script
+ import FastSort from './Xsort.vim'
+ def Test()
+ g:result = FastSort()
+ enddef
+ Test()
+ END
+ writefile(lines, 'Xscript.vim')
+
+ source Xscript.vim
+ assert_equal([4, 3, 2, 1, 0], g:result)
+
+ unlet g:result
+ delete('Xsort.vim')
+ delete('Xscript.vim')
+
+ var Funcref = function('s:RetSome')
+ assert_equal('some', Funcref())
+enddef
+
+" Check that when searching for "FilterFunc" it finds the import in the
+" script where FastFilter() is called from, both as a string and as a direct
+" function reference.
+def Test_vim9script_funcref_other_script()
+ var filterLines =<< trim END
+ vim9script
+ export def FilterFunc(idx: number, val: number): bool
+ return idx % 2 == 1
+ enddef
+ export def FastFilter(): list<number>
+ return range(10)->filter('FilterFunc')
+ enddef
+ export def FastFilterDirect(): list<number>
+ return range(10)->filter(FilterFunc)
+ enddef
+ END
+ writefile(filterLines, 'Xfilter.vim')
+
+ var lines =<< trim END
+ vim9script
+ import {FilterFunc, FastFilter, FastFilterDirect} from './Xfilter.vim'
+ def Test()
+ var x: list<number> = FastFilter()
+ enddef
+ Test()
+ def TestDirect()
+ var x: list<number> = FastFilterDirect()
+ enddef
+ TestDirect()
+ END
+ CheckScriptSuccess(lines)
+ delete('Xfilter.vim')
+enddef
+
+def Test_vim9script_reload_delfunc()
+ var first_lines =<< trim END
+ vim9script
+ def FuncYes(): string
+ return 'yes'
+ enddef
+ END
+ var withno_lines =<< trim END
+ def FuncNo(): string
+ return 'no'
+ enddef
+ def g:DoCheck(no_exists: bool)
+ assert_equal('yes', FuncYes())
+ assert_equal('no', FuncNo())
+ enddef
+ END
+ var nono_lines =<< trim END
+ def g:DoCheck(no_exists: bool)
+ assert_equal('yes', FuncYes())
+ assert_fails('FuncNo()', 'E117:', '', 2, 'DoCheck')
+ enddef
+ END
+
+ # FuncNo() is defined
+ writefile(first_lines + withno_lines, 'Xreloaded.vim')
+ source Xreloaded.vim
+ g:DoCheck(true)
+
+ # FuncNo() is not redefined
+ writefile(first_lines + nono_lines, 'Xreloaded.vim')
+ source Xreloaded.vim
+ g:DoCheck(false)
+
+ # FuncNo() is back
+ writefile(first_lines + withno_lines, 'Xreloaded.vim')
+ source Xreloaded.vim
+ g:DoCheck(false)
+
+ delete('Xreloaded.vim')
+enddef
+
+def Test_vim9script_reload_delvar()
+ # write the script with a script-local variable
+ var lines =<< trim END
+ vim9script
+ var name = 'string'
+ END
+ writefile(lines, 'XreloadVar.vim')
+ source XreloadVar.vim
+
+ # now write the script using the same variable locally - works
+ lines =<< trim END
+ vim9script
+ def Func()
+ var name = 'string'
+ enddef
+ END
+ writefile(lines, 'XreloadVar.vim')
+ source XreloadVar.vim
+
+ delete('XreloadVar.vim')
+enddef
+
+def Test_import_absolute()
+ var import_lines = [
+ 'vim9script',
+ 'import exported from "' .. escape(getcwd(), '\') .. '/Xexport_abs.vim"',
+ 'def UseExported()',
+ ' g:imported_abs = exported',
+ ' exported = 8888',
+ ' g:imported_after = exported',
+ 'enddef',
+ 'UseExported()',
+ 'g:import_disassembled = execute("disass UseExported")',
+ ]
+ writefile(import_lines, 'Ximport_abs.vim')
+ writefile(s:export_script_lines, 'Xexport_abs.vim')
+
+ source Ximport_abs.vim
+
+ assert_equal(9876, g:imported_abs)
+ assert_equal(8888, g:imported_after)
+ assert_match('<SNR>\d\+_UseExported\_s*' ..
+ 'g:imported_abs = exported\_s*' ..
+ '0 LOADSCRIPT exported-2 from .*Xexport_abs.vim\_s*' ..
+ '1 STOREG g:imported_abs\_s*' ..
+ 'exported = 8888\_s*' ..
+ '2 PUSHNR 8888\_s*' ..
+ '3 STORESCRIPT exported-2 in .*Xexport_abs.vim\_s*' ..
+ 'g:imported_after = exported\_s*' ..
+ '4 LOADSCRIPT exported-2 from .*Xexport_abs.vim\_s*' ..
+ '5 STOREG g:imported_after',
+ g:import_disassembled)
+
+ Undo_export_script_lines()
+ unlet g:imported_abs
+ unlet g:import_disassembled
+
+ delete('Ximport_abs.vim')
+ delete('Xexport_abs.vim')
+enddef
+
+def Test_import_rtp()
+ var import_lines = [
+ 'vim9script',
+ 'import exported from "Xexport_rtp.vim"',
+ 'g:imported_rtp = exported',
+ ]
+ writefile(import_lines, 'Ximport_rtp.vim')
+ mkdir('import')
+ writefile(s:export_script_lines, 'import/Xexport_rtp.vim')
+
+ var save_rtp = &rtp
+ &rtp = getcwd()
+ source Ximport_rtp.vim
+ &rtp = save_rtp
+
+ assert_equal(9876, g:imported_rtp)
+
+ Undo_export_script_lines()
+ unlet g:imported_rtp
+ delete('Ximport_rtp.vim')
+ delete('import', 'rf')
+enddef
+
+def Test_import_compile_error()
+ var export_lines = [
+ 'vim9script',
+ 'export def ExpFunc(): string',
+ ' return notDefined',
+ 'enddef',
+ ]
+ writefile(export_lines, 'Xexported.vim')
+
+ var import_lines = [
+ 'vim9script',
+ 'import ExpFunc from "./Xexported.vim"',
+ 'def ImpFunc()',
+ ' echo ExpFunc()',
+ 'enddef',
+ 'defcompile',
+ ]
+ writefile(import_lines, 'Ximport.vim')
+
+ try
+ source Ximport.vim
+ catch /E1001/
+ # Error should be fore the Xexported.vim file.
+ assert_match('E1001: Variable not found: notDefined', v:exception)
+ assert_match('function <SNR>\d\+_ImpFunc\[1\]..<SNR>\d\+_ExpFunc, line 1', v:throwpoint)
+ endtry
+
+ delete('Xexported.vim')
+ delete('Ximport.vim')
+enddef
+
+def Test_func_redefine_error()
+ var lines = [
+ 'vim9script',
+ 'def Func()',
+ ' eval [][0]',
+ 'enddef',
+ 'Func()',
+ ]
+ writefile(lines, 'Xtestscript.vim')
+
+ for count in range(3)
+ try
+ source Xtestscript.vim
+ catch /E684/
+ # function name should contain <SNR> every time
+ assert_match('E684: list index out of range', v:exception)
+ assert_match('function <SNR>\d\+_Func, line 1', v:throwpoint)
+ endtry
+ endfor
+
+ delete('Xtestscript.vim')
+enddef
+
+def Test_func_overrules_import_fails()
+ var export_lines =<< trim END
+ vim9script
+ export def Func()
+ echo 'imported'
+ enddef
+ END
+ writefile(export_lines, 'XexportedFunc.vim')
+
+ var lines =<< trim END
+ vim9script
+ import Func from './XexportedFunc.vim'
+ def Func()
+ echo 'local to function'
+ enddef
+ END
+ CheckScriptFailure(lines, 'E1073:')
+
+ lines =<< trim END
+ vim9script
+ import Func from './XexportedFunc.vim'
+ def Outer()
+ def Func()
+ echo 'local to function'
+ enddef
+ enddef
+ defcompile
+ END
+ CheckScriptFailure(lines, 'E1073:')
+
+ delete('XexportedFunc.vim')
+enddef
+
+def Test_func_redefine_fails()
+ var lines =<< trim END
+ vim9script
+ def Func()
+ echo 'one'
+ enddef
+ def Func()
+ echo 'two'
+ enddef
+ END
+ CheckScriptFailure(lines, 'E1073:')
+
+ lines =<< trim END
+ vim9script
+ def Foo(): string
+ return 'foo'
+ enddef
+ def Func()
+ var Foo = {-> 'lambda'}
+ enddef
+ defcompile
+ END
+ CheckScriptFailure(lines, 'E1073:')
+enddef
+
+def Test_fixed_size_list()
+ # will be allocated as one piece of memory, check that changes work
+ var l = [1, 2, 3, 4]
+ l->remove(0)
+ l->add(5)
+ l->insert(99, 1)
+ assert_equal([2, 99, 3, 4, 5], l)
+enddef
+
+def Test_no_insert_xit()
+ CheckDefExecFailure(['a = 1'], 'E1100:')
+ CheckDefExecFailure(['c = 1'], 'E1100:')
+ CheckDefExecFailure(['i = 1'], 'E1100:')
+ CheckDefExecFailure(['t = 1'], 'E1100:')
+ CheckDefExecFailure(['x = 1'], 'E1100:')
+
+ CheckScriptFailure(['vim9script', 'a = 1'], 'E488:')
+ CheckScriptFailure(['vim9script', 'a'], 'E1100:')
+ CheckScriptFailure(['vim9script', 'c = 1'], 'E488:')
+ CheckScriptFailure(['vim9script', 'c'], 'E1100:')
+ CheckScriptFailure(['vim9script', 'i = 1'], 'E488:')
+ CheckScriptFailure(['vim9script', 'i'], 'E1100:')
+ CheckScriptFailure(['vim9script', 't'], 'E1100:')
+ CheckScriptFailure(['vim9script', 't = 1'], 'E1100:')
+ CheckScriptFailure(['vim9script', 'x = 1'], 'E1100:')
+enddef
+
+def IfElse(what: number): string
+ var res = ''
+ if what == 1
+ res = "one"
+ elseif what == 2
+ res = "two"
+ else
+ res = "three"
+ endif
+ return res
+enddef
+
+def Test_if_elseif_else()
+ assert_equal('one', IfElse(1))
+ assert_equal('two', IfElse(2))
+ assert_equal('three', IfElse(3))
+enddef
+
+def Test_if_elseif_else_fails()
+ CheckDefFailure(['elseif true'], 'E582:')
+ CheckDefFailure(['else'], 'E581:')
+ CheckDefFailure(['endif'], 'E580:')
+ CheckDefFailure(['if g:abool', 'elseif xxx'], 'E1001:')
+ CheckDefFailure(['if true', 'echo 1'], 'E171:')
+enddef
+
+let g:bool_true = v:true
+let g:bool_false = v:false
+
+def Test_if_const_expr()
+ var res = false
+ if true ? true : false
+ res = true
+ endif
+ assert_equal(true, res)
+
+ g:glob = 2
+ if false
+ execute('g:glob = 3')
+ endif
+ assert_equal(2, g:glob)
+ if true
+ execute('g:glob = 3')
+ endif
+ assert_equal(3, g:glob)
+
+ res = false
+ if g:bool_true ? true : false
+ res = true
+ endif
+ assert_equal(true, res)
+
+ res = false
+ if true ? g:bool_true : false
+ res = true
+ endif
+ assert_equal(true, res)
+
+ res = false
+ if true ? true : g:bool_false
+ res = true
+ endif
+ assert_equal(true, res)
+
+ res = false
+ if true ? false : true
+ res = true
+ endif
+ assert_equal(false, res)
+
+ res = false
+ if false ? false : true
+ res = true
+ endif
+ assert_equal(true, res)
+
+ res = false
+ if false ? true : false
+ res = true
+ endif
+ assert_equal(false, res)
+
+ res = false
+ if has('xyz') ? true : false
+ res = true
+ endif
+ assert_equal(false, res)
+
+ res = false
+ if true && true
+ res = true
+ endif
+ assert_equal(true, res)
+
+ res = false
+ if true && false
+ res = true
+ endif
+ assert_equal(false, res)
+
+ res = false
+ if g:bool_true && false
+ res = true
+ endif
+ assert_equal(false, res)
+
+ res = false
+ if true && g:bool_false
+ res = true
+ endif
+ assert_equal(false, res)
+
+ res = false
+ if false && false
+ res = true
+ endif
+ assert_equal(false, res)
+
+ res = false
+ if true || false
+ res = true
+ endif
+ assert_equal(true, res)
+
+ res = false
+ if g:bool_true || false
+ res = true
+ endif
+ assert_equal(true, res)
+
+ res = false
+ if true || g:bool_false
+ res = true
+ endif
+ assert_equal(true, res)
+
+ res = false
+ if false || false
+ res = true
+ endif
+ assert_equal(false, res)
+
+ # with constant "false" expression may be invalid so long as the syntax is OK
+ if false | eval 0 | endif
+ if false | eval burp + 234 | endif
+ if false | echo burp 234 'asd' | endif
+ if false
+ burp
+ endif
+enddef
+
+def Test_if_const_expr_fails()
+ CheckDefFailure(['if "aaa" == "bbb'], 'E114:')
+ CheckDefFailure(["if 'aaa' == 'bbb"], 'E115:')
+ CheckDefFailure(["if has('aaa'"], 'E110:')
+ CheckDefFailure(["if has('aaa') ? true false"], 'E109:')
+enddef
+
+def RunNested(i: number): number
+ var x: number = 0
+ if i % 2
+ if 1
+ # comment
+ else
+ # comment
+ endif
+ x += 1
+ else
+ x += 1000
+ endif
+ return x
+enddef
+
+def Test_nested_if()
+ assert_equal(1, RunNested(1))
+ assert_equal(1000, RunNested(2))
+enddef
+
+def Test_execute_cmd()
+ # missing argument is ignored
+ execute
+ execute # comment
+
+ new
+ setline(1, 'default')
+ execute 'setline(1, "execute-string")'
+ assert_equal('execute-string', getline(1))
+
+ execute "setline(1, 'execute-string')"
+ assert_equal('execute-string', getline(1))
+
+ var cmd1 = 'setline(1,'
+ var cmd2 = '"execute-var")'
+ execute cmd1 cmd2 # comment
+ assert_equal('execute-var', getline(1))
+
+ execute cmd1 cmd2 '|setline(1, "execute-var-string")'
+ assert_equal('execute-var-string', getline(1))
+
+ var cmd_first = 'call '
+ var cmd_last = 'setline(1, "execute-var-var")'
+ execute cmd_first .. cmd_last
+ assert_equal('execute-var-var', getline(1))
+ bwipe!
+
+ var n = true
+ execute 'echomsg' (n ? '"true"' : '"no"')
+ assert_match('^true$', Screenline(&lines))
+
+ echomsg [1, 2, 3] {a: 1, b: 2}
+ assert_match('^\[1, 2, 3\] {''a'': 1, ''b'': 2}$', Screenline(&lines))
+
+ CheckDefFailure(['execute xxx'], 'E1001:', 1)
+ CheckDefExecFailure(['execute "tabnext " .. 8'], 'E475:', 1)
+ CheckDefFailure(['execute "cmd"# comment'], 'E488:', 1)
+enddef
+
+def Test_execute_cmd_vimscript()
+ # only checks line continuation
+ var lines =<< trim END
+ vim9script
+ execute 'g:someVar'
+ .. ' = ' ..
+ '28'
+ assert_equal(28, g:someVar)
+ unlet g:someVar
+ END
+ CheckScriptSuccess(lines)
+enddef
+
+def Test_echo_cmd()
+ echo 'some' # comment
+ echon 'thing'
+ assert_match('^something$', Screenline(&lines))
+
+ echo "some" # comment
+ echon "thing"
+ assert_match('^something$', Screenline(&lines))
+
+ var str1 = 'some'
+ var str2 = 'more'
+ echo str1 str2
+ assert_match('^some more$', Screenline(&lines))
+
+ CheckDefFailure(['echo "xxx"# comment'], 'E488:')
+enddef
+
+def Test_echomsg_cmd()
+ echomsg 'some' 'more' # comment
+ assert_match('^some more$', Screenline(&lines))
+ echo 'clear'
+ :1messages
+ assert_match('^some more$', Screenline(&lines))
+
+ CheckDefFailure(['echomsg "xxx"# comment'], 'E488:')
+enddef
+
+def Test_echomsg_cmd_vimscript()
+ # only checks line continuation
+ var lines =<< trim END
+ vim9script
+ echomsg 'here'
+ .. ' is ' ..
+ 'a message'
+ assert_match('^here is a message$', Screenline(&lines))
+ END
+ CheckScriptSuccess(lines)
+enddef
+
+def Test_echoerr_cmd()
+ try
+ echoerr 'something' 'wrong' # comment
+ catch
+ assert_match('something wrong', v:exception)
+ endtry
+enddef
+
+def Test_echoerr_cmd_vimscript()
+ # only checks line continuation
+ var lines =<< trim END
+ vim9script
+ try
+ echoerr 'this'
+ .. ' is ' ..
+ 'wrong'
+ catch
+ assert_match('this is wrong', v:exception)
+ endtry
+ END
+ CheckScriptSuccess(lines)
+enddef
+
+def Test_for_outside_of_function()
+ var lines =<< trim END
+ vim9script
+ new
+ for var in range(0, 3)
+ append(line('$'), var)
+ endfor
+ assert_equal(['', '0', '1', '2', '3'], getline(1, '$'))
+ bwipe!
+ END
+ writefile(lines, 'Xvim9for.vim')
+ source Xvim9for.vim
+ delete('Xvim9for.vim')
+enddef
+
+def Test_for_loop()
+ var result = ''
+ for cnt in range(7)
+ if cnt == 4
+ break
+ endif
+ if cnt == 2
+ continue
+ endif
+ result ..= cnt .. '_'
+ endfor
+ assert_equal('0_1_3_', result)
+
+ var concat = ''
+ for str in eval('["one", "two"]')
+ concat ..= str
+ endfor
+ assert_equal('onetwo', concat)
+
+ var total = 0
+ for nr in
+ [1, 2, 3]
+ total += nr
+ endfor
+ assert_equal(6, total)
+
+ total = 0
+ for nr
+ in [1, 2, 3]
+ total += nr
+ endfor
+ assert_equal(6, total)
+
+ total = 0
+ for nr
+ in
+ [1, 2, 3]
+ total += nr
+ endfor
+ assert_equal(6, total)
+
+ var res = ""
+ for [n: number, s: string] in [[1, 'a'], [2, 'b']]
+ res ..= n .. s
+ endfor
+ assert_equal('1a2b', res)
+enddef
+
+def Test_for_loop_fails()
+ CheckDefFailure(['for '], 'E1097:')
+ CheckDefFailure(['for x'], 'E1097:')
+ CheckDefFailure(['for x in'], 'E1097:')
+ CheckDefFailure(['for # in range(5)'], 'E690:')
+ CheckDefFailure(['for i In range(5)'], 'E690:')
+ CheckDefFailure(['var x = 5', 'for x in range(5)'], 'E1017:')
+ CheckScriptFailure(['def Func(arg: any)', 'for arg in range(5)', 'enddef', 'defcompile'], 'E1006:')
+ delfunc! g:Func
+ CheckDefFailure(['for i in "text"'], 'E1012:')
+ CheckDefFailure(['for i in xxx'], 'E1001:')
+ CheckDefFailure(['endfor'], 'E588:')
+ CheckDefFailure(['for i in range(3)', 'echo 3'], 'E170:')
+enddef
+
+def Test_for_loop_script_var()
+ # cannot use s:var in a :def function
+ CheckDefFailure(['for s:var in range(3)', 'echo 3'], 'E1101:')
+
+ # can use s:var in Vim9 script, with or without s:
+ var lines =<< trim END
+ vim9script
+ var total = 0
+ for s:var in [1, 2, 3]
+ total += s:var
+ endfor
+ assert_equal(6, total)
+
+ total = 0
+ for var in [1, 2, 3]
+ total += var
+ endfor
+ assert_equal(6, total)
+ END
+enddef
+
+def Test_for_loop_unpack()
+ var lines =<< trim END
+ var result = []
+ for [v1, v2] in [[1, 2], [3, 4]]
+ result->add(v1)
+ result->add(v2)
+ endfor
+ assert_equal([1, 2, 3, 4], result)
+
+ result = []
+ for [v1, v2; v3] in [[1, 2], [3, 4, 5, 6]]
+ result->add(v1)
+ result->add(v2)
+ result->add(v3)
+ endfor
+ assert_equal([1, 2, [], 3, 4, [5, 6]], result)
+
+ result = []
+ for [&ts, &sw] in [[1, 2], [3, 4]]
+ result->add(&ts)
+ result->add(&sw)
+ endfor
+ assert_equal([1, 2, 3, 4], result)
+
+ var slist: list<string>
+ for [$LOOPVAR, @r, v:errmsg] in [['a', 'b', 'c'], ['d', 'e', 'f']]
+ slist->add($LOOPVAR)
+ slist->add(@r)
+ slist->add(v:errmsg)
+ endfor
+ assert_equal(['a', 'b', 'c', 'd', 'e', 'f'], slist)
+
+ slist = []
+ for [g:globalvar, b:bufvar, w:winvar, t:tabvar] in [['global', 'buf', 'win', 'tab'], ['1', '2', '3', '4']]
+ slist->add(g:globalvar)
+ slist->add(b:bufvar)
+ slist->add(w:winvar)
+ slist->add(t:tabvar)
+ endfor
+ assert_equal(['global', 'buf', 'win', 'tab', '1', '2', '3', '4'], slist)
+ unlet! g:globalvar b:bufvar w:winvar t:tabvar
+ END
+ CheckDefAndScriptSuccess(lines)
+
+ lines =<< trim END
+ for [v1, v2] in [[1, 2, 3], [3, 4]]
+ echo v1 v2
+ endfor
+ END
+ CheckDefExecFailure(lines, 'E710:', 1)
+
+ lines =<< trim END
+ for [v1, v2] in [[1], [3, 4]]
+ echo v1 v2
+ endfor
+ END
+ CheckDefExecFailure(lines, 'E711:', 1)
+
+ lines =<< trim END
+ for [v1, v1] in [[1, 2], [3, 4]]
+ echo v1
+ endfor
+ END
+ CheckDefExecFailure(lines, 'E1017:', 1)
+enddef
+
+def Test_while_loop()
+ var result = ''
+ var cnt = 0
+ while cnt < 555
+ if cnt == 3
+ break
+ endif
+ cnt += 1
+ if cnt == 2
+ continue
+ endif
+ result ..= cnt .. '_'
+ endwhile
+ assert_equal('1_3_', result)
+enddef
+
+def Test_while_loop_fails()
+ CheckDefFailure(['while xxx'], 'E1001:')
+ CheckDefFailure(['endwhile'], 'E588:')
+ CheckDefFailure(['continue'], 'E586:')
+ CheckDefFailure(['if true', 'continue'], 'E586:')
+ CheckDefFailure(['break'], 'E587:')
+ CheckDefFailure(['if true', 'break'], 'E587:')
+ CheckDefFailure(['while 1', 'echo 3'], 'E170:')
+enddef
+
+def Test_interrupt_loop()
+ var caught = false
+ var x = 0
+ try
+ while 1
+ x += 1
+ if x == 100
+ feedkeys("\<C-C>", 'Lt')
+ endif
+ endwhile
+ catch
+ caught = true
+ assert_equal(100, x)
+ endtry
+ assert_true(caught, 'should have caught an exception')
+ # consume the CTRL-C
+ getchar(0)
+enddef
+
+def Test_automatic_line_continuation()
+ var mylist = [
+ 'one',
+ 'two',
+ 'three',
+ ] # comment
+ assert_equal(['one', 'two', 'three'], mylist)
+
+ var mydict = {
+ ['one']: 1,
+ ['two']: 2,
+ ['three']:
+ 3,
+ } # comment
+ assert_equal({one: 1, two: 2, three: 3}, mydict)
+ mydict = {
+ one: 1, # comment
+ two: # comment
+ 2, # comment
+ three: 3 # comment
+ }
+ assert_equal({one: 1, two: 2, three: 3}, mydict)
+ mydict = {
+ one: 1,
+ two:
+ 2,
+ three: 3
+ }
+ assert_equal({one: 1, two: 2, three: 3}, mydict)
+
+ assert_equal(
+ ['one', 'two', 'three'],
+ split('one two three')
+ )
+enddef
+
+def Test_vim9_comment()
+ CheckScriptSuccess([
+ 'vim9script',
+ '# something',
+ '#something',
+ '#{something',
+ ])
+
+ split Xfile
+ CheckScriptSuccess([
+ 'vim9script',
+ 'edit #something',
+ ])
+ CheckScriptSuccess([
+ 'vim9script',
+ 'edit #{something',
+ ])
+ close
+
+ CheckScriptFailure([
+ 'vim9script',
+ ':# something',
+ ], 'E488:')
+ CheckScriptFailure([
+ '# something',
+ ], 'E488:')
+ CheckScriptFailure([
+ ':# something',
+ ], 'E488:')
+
+ { # block start
+ } # block end
+ CheckDefFailure([
+ '{# comment',
+ ], 'E488:')
+ CheckDefFailure([
+ '{',
+ '}# comment',
+ ], 'E488:')
+
+ echo "yes" # comment
+ CheckDefFailure([
+ 'echo "yes"# comment',
+ ], 'E488:')
+ CheckScriptSuccess([
+ 'vim9script',
+ 'echo "yes" # something',
+ ])
+ CheckScriptFailure([
+ 'vim9script',
+ 'echo "yes"# something',
+ ], 'E121:')
+ CheckScriptFailure([
+ 'vim9script',
+ 'echo# something',
+ ], 'E1144:')
+ CheckScriptFailure([
+ 'echo "yes" # something',
+ ], 'E121:')
+
+ exe "echo" # comment
+ CheckDefFailure([
+ 'exe "echo"# comment',
+ ], 'E488:')
+ CheckScriptSuccess([
+ 'vim9script',
+ 'exe "echo" # something',
+ ])
+ CheckScriptFailure([
+ 'vim9script',
+ 'exe "echo"# something',
+ ], 'E121:')
+ CheckScriptFailure([
+ 'vim9script',
+ 'exe# something',
+ ], 'E1144:')
+ CheckScriptFailure([
+ 'exe "echo" # something',
+ ], 'E121:')
+
+ CheckDefFailure([
+ 'try# comment',
+ ' echo "yes"',
+ 'catch',
+ 'endtry',
+ ], 'E1144:')
+ CheckScriptFailure([
+ 'vim9script',
+ 'try# comment',
+ 'echo "yes"',
+ ], 'E1144:')
+ CheckDefFailure([
+ 'try',
+ ' throw#comment',
+ 'catch',
+ 'endtry',
+ ], 'E1144:')
+ CheckDefFailure([
+ 'try',
+ ' throw "yes"#comment',
+ 'catch',
+ 'endtry',
+ ], 'E488:')
+ CheckDefFailure([
+ 'try',
+ ' echo "yes"',
+ 'catch# comment',
+ 'endtry',
+ ], 'E1144:')
+ CheckScriptFailure([
+ 'vim9script',
+ 'try',
+ ' echo "yes"',
+ 'catch# comment',
+ 'endtry',
+ ], 'E1144:')
+ CheckDefFailure([
+ 'try',
+ ' echo "yes"',
+ 'catch /pat/# comment',
+ 'endtry',
+ ], 'E488:')
+ CheckDefFailure([
+ 'try',
+ 'echo "yes"',
+ 'catch',
+ 'endtry# comment',
+ ], 'E1144:')
+ CheckScriptFailure([
+ 'vim9script',
+ 'try',
+ ' echo "yes"',
+ 'catch',
+ 'endtry# comment',
+ ], 'E1144:')
+
+ CheckScriptSuccess([
+ 'vim9script',
+ 'hi # comment',
+ ])
+ CheckScriptFailure([
+ 'vim9script',
+ 'hi# comment',
+ ], 'E1144:')
+ CheckScriptSuccess([
+ 'vim9script',
+ 'hi Search # comment',
+ ])
+ CheckScriptFailure([
+ 'vim9script',
+ 'hi Search# comment',
+ ], 'E416:')
+ CheckScriptSuccess([
+ 'vim9script',
+ 'hi link This Search # comment',
+ ])
+ CheckScriptFailure([
+ 'vim9script',
+ 'hi link This That# comment',
+ ], 'E413:')
+ CheckScriptSuccess([
+ 'vim9script',
+ 'hi clear This # comment',
+ 'hi clear # comment',
+ ])
+ # not tested, because it doesn't give an error but a warning:
+ # hi clear This# comment',
+ CheckScriptFailure([
+ 'vim9script',
+ 'hi clear# comment',
+ ], 'E416:')
+
+ CheckScriptSuccess([
+ 'vim9script',
+ 'hi Group term=bold',
+ 'match Group /todo/ # comment',
+ ])
+ CheckScriptFailure([
+ 'vim9script',
+ 'hi Group term=bold',
+ 'match Group /todo/# comment',
+ ], 'E488:')
+ CheckScriptSuccess([
+ 'vim9script',
+ 'match # comment',
+ ])
+ CheckScriptFailure([
+ 'vim9script',
+ 'match# comment',
+ ], 'E1144:')
+ CheckScriptSuccess([
+ 'vim9script',
+ 'match none # comment',
+ ])
+ CheckScriptFailure([
+ 'vim9script',
+ 'match none# comment',
+ ], 'E475:')
+
+ CheckScriptSuccess([
+ 'vim9script',
+ 'menutrans clear # comment',
+ ])
+ CheckScriptFailure([
+ 'vim9script',
+ 'menutrans clear# comment text',
+ ], 'E474:')
+
+ CheckScriptSuccess([
+ 'vim9script',
+ 'syntax clear # comment',
+ ])
+ CheckScriptFailure([
+ 'vim9script',
+ 'syntax clear# comment text',
+ ], 'E28:')
+ CheckScriptSuccess([
+ 'vim9script',
+ 'syntax keyword Word some',
+ 'syntax clear Word # comment',
+ ])
+ CheckScriptFailure([
+ 'vim9script',
+ 'syntax keyword Word some',
+ 'syntax clear Word# comment text',
+ ], 'E28:')
+
+ CheckScriptSuccess([
+ 'vim9script',
+ 'syntax list # comment',
+ ])
+ CheckScriptFailure([
+ 'vim9script',
+ 'syntax list# comment text',
+ ], 'E28:')
+
+ CheckScriptSuccess([
+ 'vim9script',
+ 'syntax match Word /pat/ oneline # comment',
+ ])
+ CheckScriptFailure([
+ 'vim9script',
+ 'syntax match Word /pat/ oneline# comment',
+ ], 'E475:')
+
+ CheckScriptSuccess([
+ 'vim9script',
+ 'syntax keyword Word word # comm[ent',
+ ])
+ CheckScriptFailure([
+ 'vim9script',
+ 'syntax keyword Word word# comm[ent',
+ ], 'E789:')
+
+ CheckScriptSuccess([
+ 'vim9script',
+ 'syntax match Word /pat/ # comment',
+ ])
+ CheckScriptFailure([
+ 'vim9script',
+ 'syntax match Word /pat/# comment',
+ ], 'E402:')
+
+ CheckScriptSuccess([
+ 'vim9script',
+ 'syntax match Word /pat/ contains=Something # comment',
+ ])
+ CheckScriptFailure([
+ 'vim9script',
+ 'syntax match Word /pat/ contains=Something# comment',
+ ], 'E475:')
+ CheckScriptFailure([
+ 'vim9script',
+ 'syntax match Word /pat/ contains= # comment',
+ ], 'E406:')
+ CheckScriptFailure([
+ 'vim9script',
+ 'syntax match Word /pat/ contains=# comment',
+ ], 'E475:')
+
+ CheckScriptSuccess([
+ 'vim9script',
+ 'syntax region Word start=/pat/ end=/pat/ # comment',
+ ])
+ CheckScriptFailure([
+ 'vim9script',
+ 'syntax region Word start=/pat/ end=/pat/# comment',
+ ], 'E402:')
+
+ CheckScriptSuccess([
+ 'vim9script',
+ 'syntax sync # comment',
+ ])
+ CheckScriptFailure([
+ 'vim9script',
+ 'syntax sync# comment',
+ ], 'E404:')
+ CheckScriptSuccess([
+ 'vim9script',
+ 'syntax sync ccomment # comment',
+ ])
+ CheckScriptFailure([
+ 'vim9script',
+ 'syntax sync ccomment# comment',
+ ], 'E404:')
+
+ CheckScriptSuccess([
+ 'vim9script',
+ 'syntax cluster Some contains=Word # comment',
+ ])
+ CheckScriptFailure([
+ 'vim9script',
+ 'syntax cluster Some contains=Word# comment',
+ ], 'E475:')
+
+ CheckScriptSuccess([
+ 'vim9script',
+ 'command Echo echo # comment',
+ 'command Echo # comment',
+ 'delcommand Echo',
+ ])
+ CheckScriptFailure([
+ 'vim9script',
+ 'command Echo echo# comment',
+ 'Echo',
+ ], 'E1144:')
+ delcommand Echo
+
+ var curdir = getcwd()
+ CheckScriptSuccess([
+ 'command Echo cd " comment',
+ 'Echo',
+ 'delcommand Echo',
+ ])
+ CheckScriptSuccess([
+ 'vim9script',
+ 'command Echo cd # comment',
+ 'Echo',
+ 'delcommand Echo',
+ ])
+ CheckScriptFailure([
+ 'vim9script',
+ 'command Echo cd " comment',
+ 'Echo',
+ ], 'E344:')
+ delcommand Echo
+ chdir(curdir)
+
+ CheckScriptFailure([
+ 'vim9script',
+ 'command Echo# comment',
+ ], 'E182:')
+ CheckScriptFailure([
+ 'vim9script',
+ 'command Echo echo',
+ 'command Echo# comment',
+ ], 'E182:')
+ delcommand Echo
+
+ CheckScriptSuccess([
+ 'vim9script',
+ 'function # comment',
+ ])
+ CheckScriptFailure([
+ 'vim9script',
+ 'function " comment',
+ ], 'E129:')
+ CheckScriptFailure([
+ 'vim9script',
+ 'function# comment',
+ ], 'E1144:')
+ CheckScriptSuccess([
+ 'vim9script',
+ 'function CheckScriptSuccess # comment',
+ ])
+ CheckScriptFailure([
+ 'vim9script',
+ 'function CheckScriptSuccess# comment',
+ ], 'E488:')
+
+ CheckScriptSuccess([
+ 'vim9script',
+ 'func g:DeleteMeA()',
+ 'endfunc',
+ 'delfunction g:DeleteMeA # comment',
+ ])
+ CheckScriptFailure([
+ 'vim9script',
+ 'func g:DeleteMeB()',
+ 'endfunc',
+ 'delfunction g:DeleteMeB# comment',
+ ], 'E488:')
+
+ CheckScriptSuccess([
+ 'vim9script',
+ 'call execute("ls") # comment',
+ ])
+ CheckScriptFailure([
+ 'vim9script',
+ 'call execute("ls")# comment',
+ ], 'E488:')
+
+ CheckScriptFailure([
+ 'def Test() " comment',
+ 'enddef',
+ ], 'E488:')
+ CheckScriptFailure([
+ 'vim9script',
+ 'def Test() " comment',
+ 'enddef',
+ ], 'E488:')
+
+ CheckScriptSuccess([
+ 'func Test() " comment',
+ 'endfunc',
+ 'delfunc Test',
+ ])
+ CheckScriptSuccess([
+ 'vim9script',
+ 'func Test() " comment',
+ 'endfunc',
+ ])
+
+ CheckScriptSuccess([
+ 'def Test() # comment',
+ 'enddef',
+ ])
+ CheckScriptFailure([
+ 'func Test() # comment',
+ 'endfunc',
+ ], 'E488:')
+enddef
+
+def Test_vim9_comment_gui()
+ CheckCanRunGui
+
+ CheckScriptFailure([
+ 'vim9script',
+ 'gui#comment'
+ ], 'E1144:')
+ CheckScriptFailure([
+ 'vim9script',
+ 'gui -f#comment'
+ ], 'E499:')
+enddef
+
+def Test_vim9_comment_not_compiled()
+ au TabEnter *.vim g:entered = 1
+ au TabEnter *.x g:entered = 2
+
+ edit test.vim
+ doautocmd TabEnter #comment
+ assert_equal(1, g:entered)
+
+ doautocmd TabEnter f.x
+ assert_equal(2, g:entered)
+
+ g:entered = 0
+ doautocmd TabEnter f.x #comment
+ assert_equal(2, g:entered)
+
+ assert_fails('doautocmd Syntax#comment', 'E216:')
+
+ au! TabEnter
+ unlet g:entered
+
+ CheckScriptSuccess([
+ 'vim9script',
+ 'g:var = 123',
+ 'b:var = 456',
+ 'w:var = 777',
+ 't:var = 888',
+ 'unlet g:var w:var # something',
+ ])
+
+ CheckScriptFailure([
+ 'vim9script',
+ 'let var = 123',
+ ], 'E1126: Cannot use :let in Vim9 script')
+
+ CheckScriptFailure([
+ 'vim9script',
+ 'var g:var = 123',
+ ], 'E1016: Cannot declare a global variable:')
+
+ CheckScriptFailure([
+ 'vim9script',
+ 'var b:var = 123',
+ ], 'E1016: Cannot declare a buffer variable:')
+
+ CheckScriptFailure([
+ 'vim9script',
+ 'var w:var = 123',
+ ], 'E1016: Cannot declare a window variable:')
+
+ CheckScriptFailure([
+ 'vim9script',
+ 'var t:var = 123',
+ ], 'E1016: Cannot declare a tab variable:')
+
+ CheckScriptFailure([
+ 'vim9script',
+ 'var v:version = 123',
+ ], 'E1016: Cannot declare a v: variable:')
+
+ CheckScriptFailure([
+ 'vim9script',
+ 'var $VARIABLE = "text"',
+ ], 'E1016: Cannot declare an environment variable:')
+
+ CheckScriptFailure([
+ 'vim9script',
+ 'g:var = 123',
+ 'unlet g:var# comment1',
+ ], 'E108:')
+
+ CheckScriptFailure([
+ 'let g:var = 123',
+ 'unlet g:var # something',
+ ], 'E488:')
+
+ CheckScriptSuccess([
+ 'vim9script',
+ 'if 1 # comment2',
+ ' echo "yes"',
+ 'elseif 2 #comment',
+ ' echo "no"',
+ 'endif',
+ ])
+
+ CheckScriptFailure([
+ 'vim9script',
+ 'if 1# comment3',
+ ' echo "yes"',
+ 'endif',
+ ], 'E15:')
+
+ CheckScriptFailure([
+ 'vim9script',
+ 'if 0 # comment4',
+ ' echo "yes"',
+ 'elseif 2#comment',
+ ' echo "no"',
+ 'endif',
+ ], 'E15:')
+
+ CheckScriptSuccess([
+ 'vim9script',
+ 'var v = 1 # comment5',
+ ])
+
+ CheckScriptFailure([
+ 'vim9script',
+ 'var v = 1# comment6',
+ ], 'E15:')
+
+ CheckScriptSuccess([
+ 'vim9script',
+ 'new'
+ 'setline(1, ["# define pat", "last"])',
+ ':$',
+ 'dsearch /pat/ #comment',
+ 'bwipe!',
+ ])
+
+ CheckScriptFailure([
+ 'vim9script',
+ 'new'
+ 'setline(1, ["# define pat", "last"])',
+ ':$',
+ 'dsearch /pat/#comment',
+ 'bwipe!',
+ ], 'E488:')
+
+ CheckScriptFailure([
+ 'vim9script',
+ 'func! SomeFunc()',
+ ], 'E477:')
+enddef
+
+def Test_finish()
+ var lines =<< trim END
+ vim9script
+ g:res = 'one'
+ if v:false | finish | endif
+ g:res = 'two'
+ finish
+ g:res = 'three'
+ END
+ writefile(lines, 'Xfinished')
+ source Xfinished
+ assert_equal('two', g:res)
+
+ unlet g:res
+ delete('Xfinished')
+enddef
+
+def Test_forward_declaration()
+ var lines =<< trim END
+ vim9script
+ def GetValue(): string
+ return theVal
+ enddef
+ var theVal = 'something'
+ g:initVal = GetValue()
+ theVal = 'else'
+ g:laterVal = GetValue()
+ END
+ writefile(lines, 'Xforward')
+ source Xforward
+ assert_equal('something', g:initVal)
+ assert_equal('else', g:laterVal)
+
+ unlet g:initVal
+ unlet g:laterVal
+ delete('Xforward')
+enddef
+
+def Test_source_vim9_from_legacy()
+ var vim9_lines =<< trim END
+ vim9script
+ var local = 'local'
+ g:global = 'global'
+ export var exported = 'exported'
+ export def GetText(): string
+ return 'text'
+ enddef
+ END
+ writefile(vim9_lines, 'Xvim9_script.vim')
+
+ var legacy_lines =<< trim END
+ source Xvim9_script.vim
+
+ call assert_false(exists('local'))
+ call assert_false(exists('exported'))
+ call assert_false(exists('s:exported'))
+ call assert_equal('global', global)
+ call assert_equal('global', g:global)
+
+ " imported variable becomes script-local
+ import exported from './Xvim9_script.vim'
+ call assert_equal('exported', s:exported)
+ call assert_false(exists('exported'))
+
+ " imported function becomes script-local
+ import GetText from './Xvim9_script.vim'
+ call assert_equal('text', s:GetText())
+ call assert_false(exists('*GetText'))
+ END
+ writefile(legacy_lines, 'Xlegacy_script.vim')
+
+ source Xlegacy_script.vim
+ assert_equal('global', g:global)
+ unlet g:global
+
+ delete('Xlegacy_script.vim')
+ delete('Xvim9_script.vim')
+enddef
+
+func Test_vim9script_not_global()
+ " check that items defined in Vim9 script are script-local, not global
+ let vim9lines =<< trim END
+ vim9script
+ var name = 'local'
+ func TheFunc()
+ echo 'local'
+ endfunc
+ def DefFunc()
+ echo 'local'
+ enddef
+ END
+ call writefile(vim9lines, 'Xvim9script.vim')
+ source Xvim9script.vim
+ try
+ echo g:var
+ assert_report('did not fail')
+ catch /E121:/
+ " caught
+ endtry
+ try
+ call TheFunc()
+ assert_report('did not fail')
+ catch /E117:/
+ " caught
+ endtry
+ try
+ call DefFunc()
+ assert_report('did not fail')
+ catch /E117:/
+ " caught
+ endtry
+
+ call delete('Xvim9script.vim')
+endfunc
+
+def Test_vim9_copen()
+ # this was giving an error for setting w:quickfix_title
+ copen
+ quit
+enddef
+
+" test using an auto-loaded function and variable
+def Test_vim9_autoload()
+ var lines =<< trim END
+ vim9script
+ def some#gettest(): string
+ return 'test'
+ enddef
+ g:some#name = 'name'
+ END
+
+ mkdir('Xdir/autoload', 'p')
+ writefile(lines, 'Xdir/autoload/some.vim')
+ var save_rtp = &rtp
+ exe 'set rtp^=' .. getcwd() .. '/Xdir'
+
+ assert_equal('test', g:some#gettest())
+ assert_equal('name', g:some#name)
+ g:some#other = 'other'
+ assert_equal('other', g:some#other)
+
+ # upper case script name works
+ lines =<< trim END
+ vim9script
+ def Other#getOther(): string
+ return 'other'
+ enddef
+ END
+ writefile(lines, 'Xdir/autoload/Other.vim')
+ assert_equal('other', g:Other#getOther())
+
+ delete('Xdir', 'rf')
+ &rtp = save_rtp
+enddef
+
+" test using a vim9script that is auto-loaded from an autocmd
+def Test_vim9_aucmd_autoload()
+ var lines =<< trim END
+ vim9script
+ def foo#test()
+ echomsg getreg('"')
+ enddef
+ END
+
+ mkdir('Xdir/autoload', 'p')
+ writefile(lines, 'Xdir/autoload/foo.vim')
+ var save_rtp = &rtp
+ exe 'set rtp^=' .. getcwd() .. '/Xdir'
+ augroup test
+ autocmd TextYankPost * call foo#test()
+ augroup END
+
+ normal Y
+
+ augroup test
+ autocmd!
+ augroup END
+ delete('Xdir', 'rf')
+ &rtp = save_rtp
+enddef
+
+" This was causing a crash because suppress_errthrow wasn't reset.
+def Test_vim9_autoload_error()
+ var lines =<< trim END
+ vim9script
+ def crash#func()
+ try
+ for x in List()
+ endfor
+ catch
+ endtry
+ g:ok = true
+ enddef
+ fu List()
+ invalid
+ endfu
+ try
+ alsoinvalid
+ catch /wontmatch/
+ endtry
+ END
+ call mkdir('Xruntime/autoload', 'p')
+ call writefile(lines, 'Xruntime/autoload/crash.vim')
+
+ # run in a separate Vim to avoid the side effects of assert_fails()
+ lines =<< trim END
+ exe 'set rtp^=' .. getcwd() .. '/Xruntime'
+ call crash#func()
+ call writefile(['ok'], 'Xdidit')
+ qall!
+ END
+ writefile(lines, 'Xscript')
+ RunVim([], [], '-S Xscript')
+ assert_equal(['ok'], readfile('Xdidit'))
+
+ delete('Xdidit')
+ delete('Xscript')
+ delete('Xruntime', 'rf')
+
+ lines =<< trim END
+ vim9script
+ var foo#bar = 'asdf'
+ END
+ CheckScriptFailure(lines, 'E461: Illegal variable name: foo#bar', 2)
+enddef
+
+def Test_script_var_in_autocmd()
+ # using a script variable from an autocommand, defined in a :def function in a
+ # legacy Vim script, cannot check the variable type.
+ var lines =<< trim END
+ let s:counter = 1
+ def s:Func()
+ au! CursorHold
+ au CursorHold * s:counter += 1
+ enddef
+ call s:Func()
+ doau CursorHold
+ call assert_equal(2, s:counter)
+ au! CursorHold
+ END
+ CheckScriptSuccess(lines)
+enddef
+
+def Test_cmdline_win()
+ # if the Vim syntax highlighting uses Vim9 constructs they can be used from
+ # the command line window.
+ mkdir('rtp/syntax', 'p')
+ var export_lines =<< trim END
+ vim9script
+ export var That = 'yes'
+ END
+ writefile(export_lines, 'rtp/syntax/Xexport.vim')
+ var import_lines =<< trim END
+ vim9script
+ import That from './Xexport.vim'
+ END
+ writefile(import_lines, 'rtp/syntax/vim.vim')
+ var save_rtp = &rtp
+ &rtp = getcwd() .. '/rtp' .. ',' .. &rtp
+ syntax on
+ augroup CmdWin
+ autocmd CmdwinEnter * g:got_there = 'yes'
+ augroup END
+ # this will open and also close the cmdline window
+ feedkeys('q:', 'xt')
+ assert_equal('yes', g:got_there)
+
+ augroup CmdWin
+ au!
+ augroup END
+ &rtp = save_rtp
+ delete('rtp', 'rf')
+enddef
+
+def Test_invalid_sid()
+ assert_fails('func <SNR>1234_func', 'E123:')
+
+ if RunVim([], ['wq! Xdidit'], '+"func <SNR>1_func"')
+ assert_equal([], readfile('Xdidit'))
+ endif
+ delete('Xdidit')
+enddef
+
+def Test_restoring_cpo()
+ writefile(['vim9script', 'set nocp'], 'Xsourced')
+ writefile(['call writefile(["done"], "Xdone")', 'quit!'], 'Xclose')
+ if RunVim([], [], '-u NONE +"set cpo+=a" -S Xsourced -S Xclose')
+ assert_equal(['done'], readfile('Xdone'))
+ endif
+ delete('Xsourced')
+ delete('Xclose')
+ delete('Xdone')
+enddef
+
+
+def Test_unset_any_variable()
+ var lines =<< trim END
+ var name: any
+ assert_equal(0, name)
+ END
+ CheckDefAndScriptSuccess(lines)
+enddef
+
+func Test_define_func_at_command_line()
+ CheckRunVimInTerminal
+
+ " call indirectly to avoid compilation error for missing functions
+ call Run_Test_define_func_at_command_line()
+endfunc
+
+def Run_Test_define_func_at_command_line()
+ # run in a separate Vim instance to avoid the script context
+ var lines =<< trim END
+ func CheckAndQuit()
+ call assert_fails('call Afunc()', 'E117: Unknown function: Bfunc')
+ call writefile(['errors: ' .. string(v:errors)], 'Xdidcmd')
+ endfunc
+ END
+ writefile([''], 'Xdidcmd')
+ writefile(lines, 'XcallFunc')
+ var buf = RunVimInTerminal('-S XcallFunc', {rows: 6})
+ # define Afunc() on the command line
+ term_sendkeys(buf, ":def Afunc()\<CR>Bfunc()\<CR>enddef\<CR>")
+ term_sendkeys(buf, ":call CheckAndQuit()\<CR>")
+ WaitForAssert(() => assert_equal(['errors: []'], readfile('Xdidcmd')))
+
+ call StopVimInTerminal(buf)
+ delete('XcallFunc')
+ delete('Xdidcmd')
+enddef
+
+def Test_script_var_scope()
+ var lines =<< trim END
+ vim9script
+ if true
+ if true
+ var one = 'one'
+ echo one
+ endif
+ echo one
+ endif
+ END
+ CheckScriptFailure(lines, 'E121:', 7)
+
+ lines =<< trim END
+ vim9script
+ if true
+ if false
+ var one = 'one'
+ echo one
+ else
+ var one = 'one'
+ echo one
+ endif
+ echo one
+ endif
+ END
+ CheckScriptFailure(lines, 'E121:', 10)
+
+ lines =<< trim END
+ vim9script
+ while true
+ var one = 'one'
+ echo one
+ break
+ endwhile
+ echo one
+ END
+ CheckScriptFailure(lines, 'E121:', 7)
+
+ lines =<< trim END
+ vim9script
+ for i in range(1)
+ var one = 'one'
+ echo one
+ endfor
+ echo one
+ END
+ CheckScriptFailure(lines, 'E121:', 6)
+
+ lines =<< trim END
+ vim9script
+ {
+ var one = 'one'
+ assert_equal('one', one)
+ }
+ assert_false(exists('one'))
+ assert_false(exists('s:one'))
+ END
+ CheckScriptSuccess(lines)
+
+ lines =<< trim END
+ vim9script
+ {
+ var one = 'one'
+ echo one
+ }
+ echo one
+ END
+ CheckScriptFailure(lines, 'E121:', 6)
+enddef
+
+def Test_catch_exception_in_callback()
+ var lines =<< trim END
+ vim9script
+ def Callback(...l: any)
+ try
+ var x: string
+ var y: string
+ # this error should be caught with CHECKLEN
+ [x, y] = ['']
+ catch
+ g:caught = 'yes'
+ endtry
+ enddef
+ popup_menu('popup', {callback: Callback})
+ feedkeys("\r", 'xt')
+ END
+ CheckScriptSuccess(lines)
+
+ unlet g:caught
+enddef
+
+def Test_no_unknown_error_after_error()
+ if !has('unix') || !has('job')
+ throw 'Skipped: not unix of missing +job feature'
+ endif
+ var lines =<< trim END
+ vim9script
+ var source: list<number>
+ def Out_cb(...l: any)
+ eval [][0]
+ enddef
+ def Exit_cb(...l: any)
+ sleep 1m
+ source += l
+ enddef
+ var myjob = job_start('echo burp', {out_cb: Out_cb, exit_cb: Exit_cb, mode: 'raw'})
+ while job_status(myjob) == 'run'
+ sleep 10m
+ endwhile
+ sleep 10m
+ END
+ writefile(lines, 'Xdef')
+ assert_fails('so Xdef', ['E684:', 'E1012:'])
+ delete('Xdef')
+enddef
+
+def InvokeNormal()
+ exe "norm! :m+1\r"
+enddef
+
+def Test_invoke_normal_in_visual_mode()
+ xnoremap <F3> <Cmd>call <SID>InvokeNormal()<CR>
+ new
+ setline(1, ['aaa', 'bbb'])
+ feedkeys("V\<F3>", 'xt')
+ assert_equal(['bbb', 'aaa'], getline(1, 2))
+ xunmap <F3>
+enddef
+
+def Test_white_space_after_command()
+ var lines =<< trim END
+ exit_cb: Func})
+ END
+ CheckDefAndScriptFailure(lines, 'E1144:', 1)
+
+ lines =<< trim END
+ e#
+ END
+ CheckDefAndScriptFailure(lines, 'E1144:', 1)
+enddef
+
+def Test_script_var_gone_when_sourced_twice()
+ var lines =<< trim END
+ vim9script
+ if exists('g:guard')
+ finish
+ endif
+ g:guard = 1
+ var name = 'thename'
+ def g:GetName(): string
+ return name
+ enddef
+ def g:SetName(arg: string)
+ name = arg
+ enddef
+ END
+ writefile(lines, 'XscriptTwice.vim')
+ so XscriptTwice.vim
+ assert_equal('thename', g:GetName())
+ g:SetName('newname')
+ assert_equal('newname', g:GetName())
+ so XscriptTwice.vim
+ assert_fails('call g:GetName()', 'E1149:')
+ assert_fails('call g:SetName("x")', 'E1149:')
+
+ delfunc g:GetName
+ delfunc g:SetName
+ delete('XscriptTwice.vim')
+ unlet g:guard
+enddef
+
+def Test_import_gone_when_sourced_twice()
+ var exportlines =<< trim END
+ vim9script
+ if exists('g:guard')
+ finish
+ endif
+ g:guard = 1
+ export var name = 'someName'
+ END
+ writefile(exportlines, 'XexportScript.vim')
+
+ var lines =<< trim END
+ vim9script
+ import name from './XexportScript.vim'
+ def g:GetName(): string
+ return name
+ enddef
+ END
+ writefile(lines, 'XscriptImport.vim')
+ so XscriptImport.vim
+ assert_equal('someName', g:GetName())
+
+ so XexportScript.vim
+ assert_fails('call g:GetName()', 'E1149:')
+
+ delfunc g:GetName
+ delete('XexportScript.vim')
+ delete('XscriptImport.vim')
+ unlet g:guard
+enddef
+
+" Keep this last, it messes up highlighting.
+def Test_substitute_cmd()
+ new
+ setline(1, 'something')
+ :substitute(some(other(
+ assert_equal('otherthing', getline(1))
+ bwipe!
+
+ # also when the context is Vim9 script
+ var lines =<< trim END
+ vim9script
+ new
+ setline(1, 'something')
+ :substitute(some(other(
+ assert_equal('otherthing', getline(1))
+ bwipe!
+ END
+ writefile(lines, 'Xvim9lines')
+ source Xvim9lines
+
+ delete('Xvim9lines')
+enddef
+
+" vim: ts=8 sw=2 sts=2 expandtab tw=80 fdm=marker