blob: ef98583fe3c04a7e346c0c029c67589d6a978d62 [file] [log] [blame]
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001" Test various aspects of the Vim9 script language.
2
3source check.vim
4source view_util.vim
Bram Moolenaar04b12692020-05-04 23:24:44 +02005source vim9.vim
Bram Moolenaar47e7d702020-07-05 18:18:42 +02006source screendump.vim
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02007
8func Test_def_basic()
9 def SomeFunc(): string
10 return 'yes'
11 enddef
12 call assert_equal('yes', SomeFunc())
13endfunc
14
15def ReturnString(): string
16 return 'string'
17enddef
18
19def ReturnNumber(): number
20 return 123
21enddef
22
23let g:notNumber = 'string'
24
25def ReturnGlobal(): number
26 return g:notNumber
27enddef
28
29def Test_return_something()
30 assert_equal('string', ReturnString())
31 assert_equal(123, ReturnNumber())
32 assert_fails('call ReturnGlobal()', 'E1029: Expected number but got string')
33enddef
34
Bram Moolenaarefd88552020-06-18 20:50:10 +020035def Test_missing_return()
36 CheckDefFailure(['def Missing(): number',
37 ' if g:cond',
38 ' echo "no return"',
39 ' else',
40 ' return 0',
41 ' endif'
42 'enddef'], 'E1027:')
43 CheckDefFailure(['def Missing(): number',
44 ' if g:cond',
45 ' return 1',
46 ' else',
47 ' echo "no return"',
48 ' endif'
49 'enddef'], 'E1027:')
50 CheckDefFailure(['def Missing(): number',
51 ' if g:cond',
52 ' return 1',
53 ' else',
54 ' return 2',
55 ' endif'
56 ' return 3'
57 'enddef'], 'E1095:')
58enddef
59
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +020060let s:nothing = 0
61def ReturnNothing()
62 s:nothing = 1
63 if true
64 return
65 endif
66 s:nothing = 2
67enddef
68
69def Test_return_nothing()
70 ReturnNothing()
71 assert_equal(1, s:nothing)
72enddef
73
74func Increment()
75 let g:counter += 1
76endfunc
77
78def Test_call_ufunc_count()
79 g:counter = 1
80 Increment()
81 Increment()
82 Increment()
Bram Moolenaarf5be8cd2020-07-17 20:36:00 +020083 # works with and without :call
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +020084 assert_equal(4, g:counter)
85 call assert_equal(4, g:counter)
86 unlet g:counter
87enddef
88
89def MyVarargs(arg: string, ...rest: list<string>): string
90 let res = arg
91 for s in rest
92 res ..= ',' .. s
93 endfor
94 return res
95enddef
96
97def Test_call_varargs()
98 assert_equal('one', MyVarargs('one'))
99 assert_equal('one,two', MyVarargs('one', 'two'))
100 assert_equal('one,two,three', MyVarargs('one', 'two', 'three'))
101enddef
102
103def MyDefaultArgs(name = 'string'): string
104 return name
105enddef
106
Bram Moolenaare30f64b2020-07-15 19:48:20 +0200107def MyDefaultSecond(name: string, second: bool = true): string
108 return second ? name : 'none'
109enddef
110
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200111def Test_call_default_args()
112 assert_equal('string', MyDefaultArgs())
113 assert_equal('one', MyDefaultArgs('one'))
114 assert_fails('call MyDefaultArgs("one", "two")', 'E118:')
115
Bram Moolenaare30f64b2020-07-15 19:48:20 +0200116 assert_equal('test', MyDefaultSecond('test'))
117 assert_equal('test', MyDefaultSecond('test', true))
118 assert_equal('none', MyDefaultSecond('test', false))
119
Bram Moolenaar822ba242020-05-24 23:00:18 +0200120 CheckScriptFailure(['def Func(arg: number = asdf)', 'enddef', 'defcompile'], 'E1001:')
121 CheckScriptFailure(['def Func(arg: number = "text")', 'enddef', 'defcompile'], 'E1013: argument 1: type mismatch, expected number but got string')
Bram Moolenaar04b12692020-05-04 23:24:44 +0200122enddef
123
124def Test_nested_function()
125 def Nested(arg: string): string
126 return 'nested ' .. arg
127 enddef
128 assert_equal('nested function', Nested('function'))
129
Bram Moolenaar0e65d3d2020-05-05 17:53:16 +0200130 CheckDefFailure(['def Nested()', 'enddef', 'Nested(66)'], 'E118:')
131 CheckDefFailure(['def Nested(arg: string)', 'enddef', 'Nested()'], 'E119:')
132
Bram Moolenaar04b12692020-05-04 23:24:44 +0200133 CheckDefFailure(['func Nested()', 'endfunc'], 'E1086:')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200134enddef
135
136func Test_call_default_args_from_func()
137 call assert_equal('string', MyDefaultArgs())
138 call assert_equal('one', MyDefaultArgs('one'))
139 call assert_fails('call MyDefaultArgs("one", "two")', 'E118:')
140endfunc
141
142func TakesOneArg(arg)
143 echo a:arg
144endfunc
145
146def Test_call_wrong_args()
147 call CheckDefFailure(['TakesOneArg()'], 'E119:')
148 call CheckDefFailure(['TakesOneArg(11, 22)'], 'E118:')
149 call CheckDefFailure(['bufnr(xxx)'], 'E1001:')
Bram Moolenaar1c0d44f2020-05-02 19:04:58 +0200150 call CheckScriptFailure(['def Func(Ref: func(s: string))'], 'E475:')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200151enddef
152
153" Default arg and varargs
154def MyDefVarargs(one: string, two = 'foo', ...rest: list<string>): string
155 let res = one .. ',' .. two
156 for s in rest
157 res ..= ',' .. s
158 endfor
159 return res
160enddef
161
162def Test_call_def_varargs()
163 call assert_fails('call MyDefVarargs()', 'E119:')
164 assert_equal('one,foo', MyDefVarargs('one'))
165 assert_equal('one,two', MyDefVarargs('one', 'two'))
166 assert_equal('one,two,three', MyDefVarargs('one', 'two', 'three'))
Bram Moolenaar49cf7cc2020-04-07 22:45:00 +0200167 call CheckDefFailure(['MyDefVarargs("one", 22)'], 'E1013: argument 2: type mismatch, expected string but got number')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200168enddef
169
Bram Moolenaar1378fbc2020-04-11 20:50:33 +0200170let s:value = ''
171
172def FuncOneDefArg(opt = 'text')
173 s:value = opt
174enddef
175
176def FuncTwoDefArg(nr = 123, opt = 'text'): string
177 return nr .. opt
178enddef
179
180def FuncVarargs(...arg: list<string>): string
181 return join(arg, ',')
182enddef
183
184def Test_func_type_varargs()
185 let RefDefArg: func(?string)
186 RefDefArg = FuncOneDefArg
187 RefDefArg()
188 assert_equal('text', s:value)
189 RefDefArg('some')
190 assert_equal('some', s:value)
191
192 let RefDef2Arg: func(?number, ?string): string
193 RefDef2Arg = FuncTwoDefArg
194 assert_equal('123text', RefDef2Arg())
195 assert_equal('99text', RefDef2Arg(99))
196 assert_equal('77some', RefDef2Arg(77, 'some'))
197
198 call CheckDefFailure(['let RefWrong: func(string?)'], 'E1010:')
199 call CheckDefFailure(['let RefWrong: func(?string, string)'], 'E1007:')
200
201 let RefVarargs: func(...list<string>): string
202 RefVarargs = FuncVarargs
203 assert_equal('', RefVarargs())
204 assert_equal('one', RefVarargs('one'))
205 assert_equal('one,two', RefVarargs('one', 'two'))
206
207 call CheckDefFailure(['let RefWrong: func(...list<string>, string)'], 'E110:')
208 call CheckDefFailure(['let RefWrong: func(...list<string>, ?string)'], 'E110:')
209enddef
210
Bram Moolenaar0b76b422020-04-07 22:05:08 +0200211" Only varargs
212def MyVarargsOnly(...args: list<string>): string
213 return join(args, ',')
214enddef
215
216def Test_call_varargs_only()
217 assert_equal('', MyVarargsOnly())
218 assert_equal('one', MyVarargsOnly('one'))
219 assert_equal('one,two', MyVarargsOnly('one', 'two'))
220 call CheckDefFailure(['MyVarargsOnly(1)'], 'E1013: argument 1: type mismatch, expected string but got number')
221 call CheckDefFailure(['MyVarargsOnly("one", 2)'], 'E1013: argument 2: type mismatch, expected string but got number')
222enddef
223
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200224def Test_using_var_as_arg()
Bram Moolenaar822ba242020-05-24 23:00:18 +0200225 call writefile(['def Func(x: number)', 'let x = 234', 'enddef', 'defcompile'], 'Xdef')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200226 call assert_fails('so Xdef', 'E1006:')
227 call delete('Xdef')
228enddef
229
Bram Moolenaarcb2bdb12020-05-10 22:53:56 +0200230def DictArg(arg: dict<string>)
231 arg['key'] = 'value'
232enddef
233
234def ListArg(arg: list<string>)
235 arg[0] = 'value'
236enddef
237
238def Test_assign_to_argument()
Bram Moolenaarf5be8cd2020-07-17 20:36:00 +0200239 # works for dict and list
Bram Moolenaarcb2bdb12020-05-10 22:53:56 +0200240 let d: dict<string> = {}
241 DictArg(d)
242 assert_equal('value', d['key'])
243 let l: list<string> = []
244 ListArg(l)
245 assert_equal('value', l[0])
246
Bram Moolenaar822ba242020-05-24 23:00:18 +0200247 call CheckScriptFailure(['def Func(arg: number)', 'arg = 3', 'enddef', 'defcompile'], 'E1090:')
Bram Moolenaarcb2bdb12020-05-10 22:53:56 +0200248enddef
249
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200250def Test_call_func_defined_later()
Bram Moolenaar1df8b3f2020-04-23 18:13:23 +0200251 call assert_equal('one', g:DefinedLater('one'))
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200252 call assert_fails('call NotDefined("one")', 'E117:')
253enddef
254
Bram Moolenaar1df8b3f2020-04-23 18:13:23 +0200255func DefinedLater(arg)
256 return a:arg
257endfunc
258
259def Test_call_funcref()
260 assert_equal(3, g:SomeFunc('abc'))
Bram Moolenaar6c4bfe42020-07-23 18:26:30 +0200261 assert_fails('NotAFunc()', 'E117:') # comment after call
Bram Moolenaar1df8b3f2020-04-23 18:13:23 +0200262 assert_fails('g:NotAFunc()', 'E117:')
Bram Moolenaar2f1980f2020-07-22 19:30:06 +0200263
264 let lines =<< trim END
265 vim9script
266 def RetNumber(): number
267 return 123
268 enddef
269 let Funcref: func: number = function('RetNumber')
270 assert_equal(123, Funcref())
271 END
272 CheckScriptSuccess(lines)
Bram Moolenaar0f60e802020-07-22 20:16:11 +0200273
274 lines =<< trim END
275 vim9script
276 def RetNumber(): number
277 return 123
278 enddef
279 def Bar(F: func: number): number
280 return F()
281 enddef
282 let Funcref = function('RetNumber')
283 assert_equal(123, Bar(Funcref))
284 END
285 CheckScriptSuccess(lines)
Bram Moolenaarbfba8652020-07-23 20:09:10 +0200286
287 lines =<< trim END
288 vim9script
289 def UseNumber(nr: number)
290 echo nr
291 enddef
292 let Funcref: func(number) = function('UseNumber')
293 Funcref(123)
294 END
295 CheckScriptSuccess(lines)
Bram Moolenaarb8070e32020-07-23 20:56:04 +0200296
297 lines =<< trim END
298 vim9script
299 def UseNumber(nr: number)
300 echo nr
301 enddef
302 let Funcref: func(string) = function('UseNumber')
303 END
304 CheckScriptFailure(lines, 'E1013: type mismatch, expected func(string) but got func(number)')
Bram Moolenaar1df8b3f2020-04-23 18:13:23 +0200305enddef
306
307let SomeFunc = function('len')
308let NotAFunc = 'text'
309
Bram Moolenaar99aaf0c2020-04-12 14:39:53 +0200310def CombineFuncrefTypes()
Bram Moolenaarf5be8cd2020-07-17 20:36:00 +0200311 # same arguments, different return type
Bram Moolenaar99aaf0c2020-04-12 14:39:53 +0200312 let Ref1: func(bool): string
313 let Ref2: func(bool): number
314 let Ref3: func(bool): any
315 Ref3 = g:cond ? Ref1 : Ref2
316
Bram Moolenaarf5be8cd2020-07-17 20:36:00 +0200317 # different number of arguments
Bram Moolenaar99aaf0c2020-04-12 14:39:53 +0200318 let Refa1: func(bool): number
319 let Refa2: func(bool, number): number
320 let Refa3: func: number
321 Refa3 = g:cond ? Refa1 : Refa2
322
Bram Moolenaarf5be8cd2020-07-17 20:36:00 +0200323 # different argument types
Bram Moolenaar99aaf0c2020-04-12 14:39:53 +0200324 let Refb1: func(bool, string): number
325 let Refb2: func(string, number): number
326 let Refb3: func(any, any): number
327 Refb3 = g:cond ? Refb1 : Refb2
328enddef
329
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200330def FuncWithForwardCall()
Bram Moolenaar1df8b3f2020-04-23 18:13:23 +0200331 return g:DefinedEvenLater("yes")
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200332enddef
333
334def DefinedEvenLater(arg: string): string
335 return arg
336enddef
337
338def Test_error_in_nested_function()
Bram Moolenaarf5be8cd2020-07-17 20:36:00 +0200339 # Error in called function requires unwinding the call stack.
Bram Moolenaar05a55512020-07-05 15:52:19 +0200340 assert_fails('call FuncWithForwardCall()', 'E1096')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200341enddef
342
343def Test_return_type_wrong()
Bram Moolenaar822ba242020-05-24 23:00:18 +0200344 CheckScriptFailure(['def Func(): number', 'return "a"', 'enddef', 'defcompile'], 'expected number but got string')
345 CheckScriptFailure(['def Func(): string', 'return 1', 'enddef', 'defcompile'], 'expected string but got number')
Bram Moolenaar05a55512020-07-05 15:52:19 +0200346 CheckScriptFailure(['def Func(): void', 'return "a"', 'enddef', 'defcompile'], 'E1096: Returning a value in a function without a return type')
347 CheckScriptFailure(['def Func()', 'return "a"', 'enddef', 'defcompile'], 'E1096: Returning a value in a function without a return type')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200348
Bram Moolenaar822ba242020-05-24 23:00:18 +0200349 CheckScriptFailure(['def Func(): number', 'return', 'enddef', 'defcompile'], 'E1003:')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200350
351 CheckScriptFailure(['def Func(): list', 'return []', 'enddef'], 'E1008:')
352 CheckScriptFailure(['def Func(): dict', 'return {}', 'enddef'], 'E1008:')
Bram Moolenaaree4e0c12020-04-06 21:35:05 +0200353 CheckScriptFailure(['def Func()', 'return 1'], 'E1057:')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200354enddef
355
356def Test_arg_type_wrong()
357 CheckScriptFailure(['def Func3(items: list)', 'echo "a"', 'enddef'], 'E1008: Missing <type>')
Bram Moolenaaree4e0c12020-04-06 21:35:05 +0200358 CheckScriptFailure(['def Func4(...)', 'echo "a"', 'enddef'], 'E1055: Missing name after ...')
Bram Moolenaarf93c7fe2020-04-23 22:16:53 +0200359 CheckScriptFailure(['def Func5(items:string)', 'echo "a"'], 'E1069:')
Bram Moolenaar6e949782020-04-13 17:21:00 +0200360 CheckScriptFailure(['def Func5(items)', 'echo "a"'], 'E1077:')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200361enddef
362
363def Test_vim9script_call()
364 let lines =<< trim END
365 vim9script
366 let var = ''
367 def MyFunc(arg: string)
368 var = arg
369 enddef
370 MyFunc('foobar')
371 assert_equal('foobar', var)
372
373 let str = 'barfoo'
374 str->MyFunc()
375 assert_equal('barfoo', var)
376
Bram Moolenaar67979662020-06-20 22:50:47 +0200377 g:value = 'value'
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200378 g:value->MyFunc()
379 assert_equal('value', var)
380
381 let listvar = []
382 def ListFunc(arg: list<number>)
383 listvar = arg
384 enddef
385 [1, 2, 3]->ListFunc()
386 assert_equal([1, 2, 3], listvar)
387
388 let dictvar = {}
389 def DictFunc(arg: dict<number>)
390 dictvar = arg
391 enddef
392 {'a': 1, 'b': 2}->DictFunc()
393 assert_equal(#{a: 1, b: 2}, dictvar)
394 def CompiledDict()
395 {'a': 3, 'b': 4}->DictFunc()
396 enddef
397 CompiledDict()
398 assert_equal(#{a: 3, b: 4}, dictvar)
399
400 #{a: 3, b: 4}->DictFunc()
401 assert_equal(#{a: 3, b: 4}, dictvar)
402
403 ('text')->MyFunc()
404 assert_equal('text', var)
405 ("some")->MyFunc()
406 assert_equal('some', var)
Bram Moolenaare6b53242020-07-01 17:28:33 +0200407
Bram Moolenaar13e12b82020-07-24 18:47:22 +0200408 # line starting with single quote is not a mark
Bram Moolenaar3d48e252020-07-15 14:15:52 +0200409 'asdfasdf'->MyFunc()
410 assert_equal('asdfasdf', var)
411
412 def UseString()
413 'xyork'->MyFunc()
414 enddef
415 UseString()
416 assert_equal('xyork', var)
417
Bram Moolenaar13e12b82020-07-24 18:47:22 +0200418 # prepending a colon makes it a mark
419 new
420 setline(1, ['aaa', 'bbb', 'ccc'])
421 normal! 3Gmt1G
422 :'t
423 assert_equal(3, getcurpos()[1])
424 bwipe!
425
Bram Moolenaare6b53242020-07-01 17:28:33 +0200426 MyFunc(
427 'continued'
428 )
429 assert_equal('continued',
430 var
431 )
432
433 call MyFunc(
434 'more'
435 ..
436 'lines'
437 )
438 assert_equal(
439 'morelines',
440 var)
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200441 END
442 writefile(lines, 'Xcall.vim')
443 source Xcall.vim
444 delete('Xcall.vim')
445enddef
446
447def Test_vim9script_call_fail_decl()
448 let lines =<< trim END
449 vim9script
450 let var = ''
451 def MyFunc(arg: string)
452 let var = 123
453 enddef
Bram Moolenaar822ba242020-05-24 23:00:18 +0200454 defcompile
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200455 END
Bram Moolenaar6c4bfe42020-07-23 18:26:30 +0200456 CheckScriptFailure(lines, 'E1054:')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200457enddef
458
Bram Moolenaar65b95452020-07-19 14:03:09 +0200459def Test_vim9script_call_fail_type()
460 let lines =<< trim END
461 vim9script
462 def MyFunc(arg: string)
463 echo arg
464 enddef
465 MyFunc(1234)
466 END
467 CheckScriptFailure(lines, 'E1013: type mismatch, expected string but got number')
468enddef
469
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200470def Test_vim9script_call_fail_const()
471 let lines =<< trim END
472 vim9script
473 const var = ''
474 def MyFunc(arg: string)
475 var = 'asdf'
476 enddef
Bram Moolenaar822ba242020-05-24 23:00:18 +0200477 defcompile
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200478 END
479 writefile(lines, 'Xcall_const.vim')
480 assert_fails('source Xcall_const.vim', 'E46:')
481 delete('Xcall_const.vim')
482enddef
483
484" Test that inside :function a Python function can be defined, :def is not
485" recognized.
486func Test_function_python()
487 CheckFeature python3
488 let py = 'python3'
489 execute py "<< EOF"
490def do_something():
491 return 1
492EOF
493endfunc
494
495def Test_delfunc()
496 let lines =<< trim END
497 vim9script
Bram Moolenaar4c17ad92020-04-27 22:47:51 +0200498 def g:GoneSoon()
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200499 echo 'hello'
500 enddef
501
502 def CallGoneSoon()
503 GoneSoon()
504 enddef
Bram Moolenaar822ba242020-05-24 23:00:18 +0200505 defcompile
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200506
Bram Moolenaar4c17ad92020-04-27 22:47:51 +0200507 delfunc g:GoneSoon
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200508 CallGoneSoon()
509 END
510 writefile(lines, 'XToDelFunc')
511 assert_fails('so XToDelFunc', 'E933')
512 assert_fails('so XToDelFunc', 'E933')
513
514 delete('XToDelFunc')
515enddef
516
517def Test_redef_failure()
518 call writefile(['def Func0(): string', 'return "Func0"', 'enddef'], 'Xdef')
519 so Xdef
520 call writefile(['def Func1(): string', 'return "Func1"', 'enddef'], 'Xdef')
521 so Xdef
Bram Moolenaar822ba242020-05-24 23:00:18 +0200522 call writefile(['def! Func0(): string', 'enddef', 'defcompile'], 'Xdef')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200523 call assert_fails('so Xdef', 'E1027:')
524 call writefile(['def Func2(): string', 'return "Func2"', 'enddef'], 'Xdef')
525 so Xdef
526 call delete('Xdef')
527
Bram Moolenaar1df8b3f2020-04-23 18:13:23 +0200528 call assert_equal(0, g:Func0())
529 call assert_equal('Func1', g:Func1())
530 call assert_equal('Func2', g:Func2())
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200531
532 delfunc! Func0
533 delfunc! Func1
534 delfunc! Func2
535enddef
536
Bram Moolenaarf93c7fe2020-04-23 22:16:53 +0200537def Test_vim9script_func()
538 let lines =<< trim END
539 vim9script
540 func Func(arg)
541 echo a:arg
542 endfunc
543 Func('text')
544 END
545 writefile(lines, 'XVim9Func')
546 so XVim9Func
547
548 delete('XVim9Func')
549enddef
550
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200551" Test for internal functions returning different types
552func Test_InternalFuncRetType()
553 let lines =<< trim END
554 def RetFloat(): float
555 return ceil(1.456)
556 enddef
557
558 def RetListAny(): list<any>
559 return items({'k' : 'v'})
560 enddef
561
562 def RetListString(): list<string>
563 return split('a:b:c', ':')
564 enddef
565
566 def RetListDictAny(): list<dict<any>>
567 return getbufinfo()
568 enddef
569
570 def RetDictNumber(): dict<number>
571 return wordcount()
572 enddef
573
574 def RetDictString(): dict<string>
575 return environ()
576 enddef
577 END
578 call writefile(lines, 'Xscript')
579 source Xscript
580
581 call assert_equal(2.0, RetFloat())
582 call assert_equal([['k', 'v']], RetListAny())
583 call assert_equal(['a', 'b', 'c'], RetListString())
584 call assert_notequal([], RetListDictAny())
585 call assert_notequal({}, RetDictNumber())
586 call assert_notequal({}, RetDictString())
587 call delete('Xscript')
588endfunc
589
590" Test for passing too many or too few arguments to internal functions
591func Test_internalfunc_arg_error()
592 let l =<< trim END
593 def! FArgErr(): float
594 return ceil(1.1, 2)
595 enddef
Bram Moolenaar822ba242020-05-24 23:00:18 +0200596 defcompile
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200597 END
598 call writefile(l, 'Xinvalidarg')
599 call assert_fails('so Xinvalidarg', 'E118:')
600 let l =<< trim END
601 def! FArgErr(): float
602 return ceil()
603 enddef
Bram Moolenaar822ba242020-05-24 23:00:18 +0200604 defcompile
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200605 END
606 call writefile(l, 'Xinvalidarg')
607 call assert_fails('so Xinvalidarg', 'E119:')
608 call delete('Xinvalidarg')
609endfunc
610
611let s:funcResult = 0
612
613def FuncNoArgNoRet()
614 funcResult = 11
615enddef
616
617def FuncNoArgRetNumber(): number
618 funcResult = 22
619 return 1234
620enddef
621
Bram Moolenaarec5929d2020-04-07 20:53:39 +0200622def FuncNoArgRetString(): string
623 funcResult = 45
624 return 'text'
625enddef
626
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200627def FuncOneArgNoRet(arg: number)
628 funcResult = arg
629enddef
630
631def FuncOneArgRetNumber(arg: number): number
632 funcResult = arg
633 return arg
634enddef
635
Bram Moolenaar08938ee2020-04-11 23:17:17 +0200636def FuncTwoArgNoRet(one: bool, two: number)
637 funcResult = two
638enddef
639
Bram Moolenaarec5929d2020-04-07 20:53:39 +0200640def FuncOneArgRetString(arg: string): string
641 return arg
642enddef
643
Bram Moolenaar89228602020-04-05 22:14:54 +0200644def FuncOneArgRetAny(arg: any): any
645 return arg
646enddef
647
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200648def Test_func_type()
649 let Ref1: func()
650 funcResult = 0
651 Ref1 = FuncNoArgNoRet
652 Ref1()
653 assert_equal(11, funcResult)
Bram Moolenaar4c683752020-04-05 21:38:23 +0200654
655 let Ref2: func
656 funcResult = 0
657 Ref2 = FuncNoArgNoRet
658 Ref2()
659 assert_equal(11, funcResult)
660
661 funcResult = 0
662 Ref2 = FuncOneArgNoRet
663 Ref2(12)
664 assert_equal(12, funcResult)
665
666 funcResult = 0
667 Ref2 = FuncNoArgRetNumber
668 assert_equal(1234, Ref2())
669 assert_equal(22, funcResult)
670
671 funcResult = 0
672 Ref2 = FuncOneArgRetNumber
673 assert_equal(13, Ref2(13))
674 assert_equal(13, funcResult)
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200675enddef
676
Bram Moolenaar9978d472020-07-05 16:01:56 +0200677def Test_repeat_return_type()
678 let res = 0
679 for n in repeat([1], 3)
680 res += n
681 endfor
682 assert_equal(3, res)
Bram Moolenaarfce82b32020-07-05 16:07:21 +0200683
684 res = 0
685 for n in add([1, 2], 3)
686 res += n
687 endfor
688 assert_equal(6, res)
Bram Moolenaar9978d472020-07-05 16:01:56 +0200689enddef
690
Bram Moolenaar846178a2020-07-05 17:04:13 +0200691def Test_argv_return_type()
692 next fileone filetwo
693 let res = ''
694 for name in argv()
695 res ..= name
696 endfor
697 assert_equal('fileonefiletwo', res)
698enddef
699
Bram Moolenaarec5929d2020-04-07 20:53:39 +0200700def Test_func_type_part()
701 let RefVoid: func: void
702 RefVoid = FuncNoArgNoRet
703 RefVoid = FuncOneArgNoRet
704 CheckDefFailure(['let RefVoid: func: void', 'RefVoid = FuncNoArgRetNumber'], 'E1013: type mismatch, expected func() but got func(): number')
Bram Moolenaar6ff71d82020-05-24 23:45:24 +0200705 CheckDefFailure(['let RefVoid: func: void', 'RefVoid = FuncNoArgRetString'], 'E1013: type mismatch, expected func() but got func(): string')
Bram Moolenaarec5929d2020-04-07 20:53:39 +0200706
707 let RefAny: func(): any
708 RefAny = FuncNoArgRetNumber
709 RefAny = FuncNoArgRetString
710 CheckDefFailure(['let RefAny: func(): any', 'RefAny = FuncNoArgNoRet'], 'E1013: type mismatch, expected func(): any but got func()')
711 CheckDefFailure(['let RefAny: func(): any', 'RefAny = FuncOneArgNoRet'], 'E1013: type mismatch, expected func(): any but got func(number)')
712
713 let RefNr: func: number
714 RefNr = FuncNoArgRetNumber
715 RefNr = FuncOneArgRetNumber
716 CheckDefFailure(['let RefNr: func: number', 'RefNr = FuncNoArgNoRet'], 'E1013: type mismatch, expected func(): number but got func()')
Bram Moolenaar6ff71d82020-05-24 23:45:24 +0200717 CheckDefFailure(['let RefNr: func: number', 'RefNr = FuncNoArgRetString'], 'E1013: type mismatch, expected func(): number but got func(): string')
Bram Moolenaarec5929d2020-04-07 20:53:39 +0200718
719 let RefStr: func: string
720 RefStr = FuncNoArgRetString
721 RefStr = FuncOneArgRetString
722 CheckDefFailure(['let RefStr: func: string', 'RefStr = FuncNoArgNoRet'], 'E1013: type mismatch, expected func(): string but got func()')
723 CheckDefFailure(['let RefStr: func: string', 'RefStr = FuncNoArgRetNumber'], 'E1013: type mismatch, expected func(): string but got func(): number')
724enddef
725
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200726def Test_func_type_fails()
727 CheckDefFailure(['let ref1: func()'], 'E704:')
728
729 CheckDefFailure(['let Ref1: func()', 'Ref1 = FuncNoArgRetNumber'], 'E1013: type mismatch, expected func() but got func(): number')
730 CheckDefFailure(['let Ref1: func()', 'Ref1 = FuncOneArgNoRet'], 'E1013: type mismatch, expected func() but got func(number)')
731 CheckDefFailure(['let Ref1: func()', 'Ref1 = FuncOneArgRetNumber'], 'E1013: type mismatch, expected func() but got func(number): number')
Bram Moolenaar6ff71d82020-05-24 23:45:24 +0200732 CheckDefFailure(['let Ref1: func(bool)', 'Ref1 = FuncTwoArgNoRet'], 'E1013: type mismatch, expected func(bool) but got func(bool, number)')
733 CheckDefFailure(['let Ref1: func(?bool)', 'Ref1 = FuncTwoArgNoRet'], 'E1013: type mismatch, expected func(?bool) but got func(bool, number)')
734 CheckDefFailure(['let Ref1: func(...bool)', 'Ref1 = FuncTwoArgNoRet'], 'E1013: type mismatch, expected func(...bool) but got func(bool, number)')
Bram Moolenaar08938ee2020-04-11 23:17:17 +0200735
736 call CheckDefFailure(['let RefWrong: func(string ,number)'], 'E1068:')
737 call CheckDefFailure(['let RefWrong: func(string,number)'], 'E1069:')
738 call CheckDefFailure(['let RefWrong: func(bool, bool, bool, bool, bool, bool, bool, bool, bool, bool, bool, bool, bool, bool, bool, bool, bool, bool, bool, bool)'], 'E740:')
739 call CheckDefFailure(['let RefWrong: func(bool):string'], 'E1069:')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200740enddef
741
Bram Moolenaar89228602020-04-05 22:14:54 +0200742def Test_func_return_type()
743 let nr: number
744 nr = FuncNoArgRetNumber()
745 assert_equal(1234, nr)
746
747 nr = FuncOneArgRetAny(122)
748 assert_equal(122, nr)
749
750 let str: string
751 str = FuncOneArgRetAny('yes')
752 assert_equal('yes', str)
753
754 CheckDefFailure(['let str: string', 'str = FuncNoArgRetNumber()'], 'E1013: type mismatch, expected string but got number')
755enddef
756
Bram Moolenaar5e774c72020-04-12 21:53:00 +0200757def MultiLine(
758 arg1: string,
759 arg2 = 1234,
760 ...rest: list<string>
761 ): string
762 return arg1 .. arg2 .. join(rest, '-')
763enddef
764
Bram Moolenaar2c330432020-04-13 14:41:35 +0200765def MultiLineComment(
766 arg1: string, # comment
767 arg2 = 1234, # comment
768 ...rest: list<string> # comment
769 ): string # comment
770 return arg1 .. arg2 .. join(rest, '-')
771enddef
772
Bram Moolenaar5e774c72020-04-12 21:53:00 +0200773def Test_multiline()
774 assert_equal('text1234', MultiLine('text'))
775 assert_equal('text777', MultiLine('text', 777))
776 assert_equal('text777one', MultiLine('text', 777, 'one'))
777 assert_equal('text777one-two', MultiLine('text', 777, 'one', 'two'))
778enddef
779
Bram Moolenaar23e03252020-04-12 22:22:31 +0200780func Test_multiline_not_vim9()
781 call assert_equal('text1234', MultiLine('text'))
782 call assert_equal('text777', MultiLine('text', 777))
783 call assert_equal('text777one', MultiLine('text', 777, 'one'))
784 call assert_equal('text777one-two', MultiLine('text', 777, 'one', 'two'))
785endfunc
786
Bram Moolenaar5e774c72020-04-12 21:53:00 +0200787
Bram Moolenaaree4e0c12020-04-06 21:35:05 +0200788" When using CheckScriptFailure() for the below test, E1010 is generated instead
789" of E1056.
790func Test_E1056_1059()
791 let caught_1056 = 0
792 try
793 def F():
794 return 1
795 enddef
796 catch /E1056:/
797 let caught_1056 = 1
798 endtry
799 call assert_equal(1, caught_1056)
800
801 let caught_1059 = 0
802 try
803 def F5(items : list)
804 echo 'a'
805 enddef
806 catch /E1059:/
807 let caught_1059 = 1
808 endtry
809 call assert_equal(1, caught_1059)
810endfunc
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200811
Bram Moolenaar015f4262020-05-05 21:25:22 +0200812func DelMe()
813 echo 'DelMe'
814endfunc
815
816def Test_deleted_function()
817 CheckDefExecFailure([
818 'let RefMe: func = function("g:DelMe")',
819 'delfunc g:DelMe',
820 'echo RefMe()'], 'E117:')
821enddef
822
823def Test_unknown_function()
824 CheckDefExecFailure([
825 'let Ref: func = function("NotExist")',
Bram Moolenaar9b7bf9e2020-07-11 22:14:59 +0200826 'delfunc g:NotExist'], 'E700:')
Bram Moolenaar015f4262020-05-05 21:25:22 +0200827enddef
828
Bram Moolenaarc8cd2b32020-05-01 19:29:08 +0200829def RefFunc(Ref: func(string): string): string
830 return Ref('more')
831enddef
832
833def Test_closure_simple()
834 let local = 'some '
835 assert_equal('some more', RefFunc({s -> local .. s}))
836enddef
837
Bram Moolenaarbf67ea12020-05-02 17:52:42 +0200838def MakeRef()
839 let local = 'some '
840 g:Ref = {s -> local .. s}
841enddef
842
843def Test_closure_ref_after_return()
844 MakeRef()
845 assert_equal('some thing', g:Ref('thing'))
846 unlet g:Ref
847enddef
848
Bram Moolenaar5adc55c2020-05-02 23:12:58 +0200849def MakeTwoRefs()
850 let local = ['some']
851 g:Extend = {s -> local->add(s)}
852 g:Read = {-> local}
853enddef
854
855def Test_closure_two_refs()
856 MakeTwoRefs()
857 assert_equal('some', join(g:Read(), ' '))
858 g:Extend('more')
859 assert_equal('some more', join(g:Read(), ' '))
860 g:Extend('even')
861 assert_equal('some more even', join(g:Read(), ' '))
862
863 unlet g:Extend
864 unlet g:Read
865enddef
866
Bram Moolenaar5adc55c2020-05-02 23:12:58 +0200867def ReadRef(Ref: func(): list<string>): string
868 return join(Ref(), ' ')
869enddef
870
871def ExtendRef(Ref: func(string), add: string)
872 Ref(add)
873enddef
874
875def Test_closure_two_indirect_refs()
Bram Moolenaarf7779c62020-05-03 15:38:16 +0200876 MakeTwoRefs()
Bram Moolenaar5adc55c2020-05-02 23:12:58 +0200877 assert_equal('some', ReadRef(g:Read))
878 ExtendRef(g:Extend, 'more')
879 assert_equal('some more', ReadRef(g:Read))
880 ExtendRef(g:Extend, 'even')
881 assert_equal('some more even', ReadRef(g:Read))
882
883 unlet g:Extend
884 unlet g:Read
885enddef
Bram Moolenaarbf67ea12020-05-02 17:52:42 +0200886
Bram Moolenaar2fd4cd72020-05-03 22:30:49 +0200887def MakeArgRefs(theArg: string)
888 let local = 'loc_val'
889 g:UseArg = {s -> theArg .. '/' .. local .. '/' .. s}
890enddef
891
892def MakeArgRefsVarargs(theArg: string, ...rest: list<string>)
893 let local = 'the_loc'
894 g:UseVararg = {s -> theArg .. '/' .. local .. '/' .. s .. '/' .. join(rest)}
895enddef
896
897def Test_closure_using_argument()
898 MakeArgRefs('arg_val')
899 assert_equal('arg_val/loc_val/call_val', g:UseArg('call_val'))
900
901 MakeArgRefsVarargs('arg_val', 'one', 'two')
902 assert_equal('arg_val/the_loc/call_val/one two', g:UseVararg('call_val'))
903
904 unlet g:UseArg
905 unlet g:UseVararg
906enddef
907
Bram Moolenaarb68b3462020-05-06 21:06:30 +0200908def MakeGetAndAppendRefs()
909 let local = 'a'
910
911 def Append(arg: string)
912 local ..= arg
913 enddef
914 g:Append = Append
915
916 def Get(): string
917 return local
918 enddef
919 g:Get = Get
920enddef
921
922def Test_closure_append_get()
923 MakeGetAndAppendRefs()
924 assert_equal('a', g:Get())
925 g:Append('-b')
926 assert_equal('a-b', g:Get())
927 g:Append('-c')
928 assert_equal('a-b-c', g:Get())
929
930 unlet g:Append
931 unlet g:Get
932enddef
933
Bram Moolenaar04b12692020-05-04 23:24:44 +0200934def Test_nested_closure()
935 let local = 'text'
936 def Closure(arg: string): string
937 return local .. arg
938 enddef
939 assert_equal('text!!!', Closure('!!!'))
940enddef
941
Bram Moolenaar6f5b6df2020-05-16 21:20:12 +0200942func GetResult(Ref)
943 return a:Ref('some')
944endfunc
945
946def Test_call_closure_not_compiled()
947 let text = 'text'
948 g:Ref = {s -> s .. text}
949 assert_equal('sometext', GetResult(g:Ref))
950enddef
951
Bram Moolenaar865af6b2020-06-18 18:45:49 +0200952def Test_sort_return_type()
953 let res: list<number>
954 res = [1, 2, 3]->sort()
955enddef
956
Bram Moolenaarf151ad12020-06-30 13:38:01 +0200957def Test_getqflist_return_type()
958 let l = getqflist()
959 assert_equal([], l)
960
961 let d = getqflist(#{items: 0})
962 assert_equal(#{items: []}, d)
963enddef
964
965def Test_getloclist_return_type()
966 let l = getloclist(1)
967 assert_equal([], l)
968
969 let d = getloclist(1, #{items: 0})
970 assert_equal(#{items: []}, d)
971enddef
972
Bram Moolenaara66ba012020-07-05 18:41:08 +0200973def Test_copy_return_type()
974 let l = copy([1, 2, 3])
975 let res = 0
976 for n in l
977 res += n
978 endfor
979 assert_equal(6, res)
980
981 let dl = deepcopy([1, 2, 3])
982 res = 0
983 for n in dl
984 res += n
985 endfor
986 assert_equal(6, res)
987enddef
988
Bram Moolenaarb3c019c2020-07-05 20:08:39 +0200989def Test_extend_return_type()
990 let l = extend([1, 2], [3])
991 let res = 0
992 for n in l
993 res += n
994 endfor
995 assert_equal(6, res)
996enddef
997
Bram Moolenaar252e88a2020-07-05 20:47:18 +0200998def Test_insert_return_type()
999 let l = insert([2, 1], 3)
1000 let res = 0
1001 for n in l
1002 res += n
1003 endfor
1004 assert_equal(6, res)
1005enddef
1006
Bram Moolenaar67627352020-07-05 21:10:24 +02001007def Test_reverse_return_type()
1008 let l = reverse([1, 2, 3])
1009 let res = 0
1010 for n in l
1011 res += n
1012 endfor
1013 assert_equal(6, res)
1014enddef
1015
Bram Moolenaarad7c2492020-07-05 20:55:29 +02001016def Test_remove_return_type()
1017 let l = remove(#{one: [1, 2], two: [3, 4]}, 'one')
1018 let res = 0
1019 for n in l
1020 res += n
1021 endfor
1022 assert_equal(3, res)
1023enddef
1024
Bram Moolenaar0d94ad62020-07-05 20:16:41 +02001025def Test_filter_return_type()
1026 let l = filter([1, 2, 3], {-> 1})
1027 let res = 0
1028 for n in l
1029 res += n
1030 endfor
1031 assert_equal(6, res)
1032enddef
1033
Bram Moolenaarf1a23682020-07-13 18:55:48 +02001034def Wrong_dict_key_type(items: list<number>): list<number>
1035 return filter(items, {_, val -> get({val: 1}, 'x')})
1036enddef
1037
1038def Test_wrong_dict_key_type()
1039 assert_fails('Wrong_dict_key_type([1, 2, 3])', 'E1029:')
1040enddef
1041
Bram Moolenaaracd4c5e2020-06-22 19:39:03 +02001042def Line_continuation_in_def(dir: string = ''): string
1043 let path: string = empty(dir)
1044 \ ? 'empty'
1045 \ : 'full'
1046 return path
1047enddef
1048
1049def Test_line_continuation_in_def()
1050 assert_equal('full', Line_continuation_in_def('.'))
1051enddef
1052
Bram Moolenaar7a4b8982020-07-08 17:36:21 +02001053def Line_continuation_in_lambda(): list<number>
1054 let x = range(97, 100)
Bram Moolenaar914e7ea2020-07-11 15:20:48 +02001055 ->map({_, v -> nr2char(v)
Bram Moolenaar7a4b8982020-07-08 17:36:21 +02001056 ->toupper()})
1057 ->reverse()
1058 return x
1059enddef
1060
1061def Test_line_continuation_in_lambda()
1062 assert_equal(['D', 'C', 'B', 'A'], Line_continuation_in_lambda())
1063enddef
1064
Bram Moolenaar8f510af2020-07-05 18:48:23 +02001065func Test_silent_echo()
Bram Moolenaar47e7d702020-07-05 18:18:42 +02001066 CheckScreendump
1067
1068 let lines =<< trim END
1069 vim9script
1070 def EchoNothing()
1071 silent echo ''
1072 enddef
1073 defcompile
1074 END
Bram Moolenaar8f510af2020-07-05 18:48:23 +02001075 call writefile(lines, 'XTest_silent_echo')
Bram Moolenaar47e7d702020-07-05 18:18:42 +02001076
1077 " Check that the balloon shows up after a mouse move
1078 let buf = RunVimInTerminal('-S XTest_silent_echo', {'rows': 6})
Bram Moolenaar8f510af2020-07-05 18:48:23 +02001079 call term_sendkeys(buf, ":abc")
Bram Moolenaar47e7d702020-07-05 18:18:42 +02001080 call VerifyScreenDump(buf, 'Test_vim9_silent_echo', {})
1081
1082 " clean up
1083 call StopVimInTerminal(buf)
1084 call delete('XTest_silent_echo')
Bram Moolenaar8f510af2020-07-05 18:48:23 +02001085endfunc
Bram Moolenaar47e7d702020-07-05 18:18:42 +02001086
Bram Moolenaar985116a2020-07-12 17:31:09 +02001087def Fibonacci(n: number): number
1088 if n < 2
1089 return n
1090 else
1091 return Fibonacci(n - 1) + Fibonacci(n - 2)
1092 endif
1093enddef
1094
1095def Test_recursive_call()
1096 assert_equal(6765, Fibonacci(20))
1097enddef
1098
Bram Moolenaar08f7a412020-07-13 20:41:08 +02001099def TreeWalk(dir: string): list<any>
1100 return readdir(dir)->map({_, val ->
1101 fnamemodify(dir .. '/' .. val, ':p')->isdirectory()
1102 ? {val : TreeWalk(dir .. '/' .. val)}
1103 : val
1104 })
1105enddef
1106
1107def Test_closure_in_map()
1108 mkdir('XclosureDir/tdir', 'p')
1109 writefile(['111'], 'XclosureDir/file1')
1110 writefile(['222'], 'XclosureDir/file2')
1111 writefile(['333'], 'XclosureDir/tdir/file3')
1112
1113 assert_equal(['file1', 'file2', {'tdir': ['file3']}], TreeWalk('XclosureDir'))
1114
1115 delete('XclosureDir', 'rf')
1116enddef
1117
Bram Moolenaara90afb92020-07-15 22:38:56 +02001118def Test_partial_call()
1119 let Xsetlist = function('setloclist', [0])
1120 Xsetlist([], ' ', {'title': 'test'})
1121 assert_equal({'title': 'test'}, getloclist(0, {'title': 1}))
1122
1123 Xsetlist = function('setloclist', [0, [], ' '])
1124 Xsetlist({'title': 'test'})
1125 assert_equal({'title': 'test'}, getloclist(0, {'title': 1}))
1126
1127 Xsetlist = function('setqflist')
1128 Xsetlist([], ' ', {'title': 'test'})
1129 assert_equal({'title': 'test'}, getqflist({'title': 1}))
1130
1131 Xsetlist = function('setqflist', [[], ' '])
1132 Xsetlist({'title': 'test'})
1133 assert_equal({'title': 'test'}, getqflist({'title': 1}))
1134enddef
1135
Bram Moolenaarf7779c62020-05-03 15:38:16 +02001136
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001137" vim: ts=8 sw=2 sts=2 expandtab tw=80 fdm=marker