blob: 86b0763dcc8b347cdb5d6ec6248ca94d112d62db [file] [log] [blame]
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001" Test various aspects of the Vim9 script language.
2
3source check.vim
Bram Moolenaarad304702020-09-06 18:22:53 +02004source term_util.vim
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02005source view_util.vim
Bram Moolenaar62aec932022-01-29 21:45:34 +00006import './vim9.vim' as v9
Bram Moolenaar47e7d702020-07-05 18:18:42 +02007source screendump.vim
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02008
9func Test_def_basic()
10 def SomeFunc(): string
11 return 'yes'
12 enddef
Bram Moolenaarc0c71e92020-09-11 19:09:48 +020013 call SomeFunc()->assert_equal('yes')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +020014endfunc
15
Bram Moolenaar2b9b17e2020-10-13 18:38:11 +020016func Test_compiling_error()
17 " use a terminal to see the whole error message
Bram Moolenaarf4e8cdd2020-10-12 22:07:13 +020018 CheckRunVimInTerminal
19
Bram Moolenaar2b9b17e2020-10-13 18:38:11 +020020 call TestCompilingError()
Bram Moolenaare8c46602021-04-05 22:27:37 +020021 call TestCompilingErrorInTry()
Bram Moolenaar2b9b17e2020-10-13 18:38:11 +020022endfunc
23
24def TestCompilingError()
Bram Moolenaarf4e8cdd2020-10-12 22:07:13 +020025 var lines =<< trim END
26 vim9script
27 def Fails()
28 echo nothing
29 enddef
30 defcompile
31 END
Bram Moolenaare8c46602021-04-05 22:27:37 +020032 writefile(lines, 'XTest_compile_error')
Bram Moolenaar62aec932022-01-29 21:45:34 +000033 var buf = g:RunVimInTerminal('-S XTest_compile_error',
Bram Moolenaare0de1712020-12-02 17:36:54 +010034 {rows: 10, wait_for_ruler: 0})
Bram Moolenaar62aec932022-01-29 21:45:34 +000035 g:WaitForAssert(() => assert_match('Error detected while compiling command line.*Fails.*Variable not found: nothing',
36 g:Term_getlines(buf, range(1, 9))))
Bram Moolenaarf4e8cdd2020-10-12 22:07:13 +020037
38 # clean up
Bram Moolenaar62aec932022-01-29 21:45:34 +000039 g:StopVimInTerminal(buf)
Bram Moolenaare8c46602021-04-05 22:27:37 +020040 delete('XTest_compile_error')
41enddef
42
43def TestCompilingErrorInTry()
44 var dir = 'Xdir/autoload'
45 mkdir(dir, 'p')
46
47 var lines =<< trim END
48 vim9script
Bram Moolenaard8fe6d32022-01-30 18:40:44 +000049 export def OnlyCompiled()
Bram Moolenaare8c46602021-04-05 22:27:37 +020050 g:runtime = 'yes'
51 invalid
52 enddef
53 END
54 writefile(lines, dir .. '/script.vim')
55
56 lines =<< trim END
57 vim9script
58 todo
59 try
60 script#OnlyCompiled()
61 catch /nothing/
62 endtry
63 END
64 lines[1] = 'set rtp=' .. getcwd() .. '/Xdir'
65 writefile(lines, 'XTest_compile_error')
66
Bram Moolenaar62aec932022-01-29 21:45:34 +000067 var buf = g:RunVimInTerminal('-S XTest_compile_error', {rows: 10, wait_for_ruler: 0})
68 g:WaitForAssert(() => assert_match('Error detected while compiling command line.*function script#OnlyCompiled.*Invalid command: invalid',
69 g:Term_getlines(buf, range(1, 9))))
Bram Moolenaare8c46602021-04-05 22:27:37 +020070
71 # clean up
Bram Moolenaar62aec932022-01-29 21:45:34 +000072 g:StopVimInTerminal(buf)
Bram Moolenaare8c46602021-04-05 22:27:37 +020073 delete('XTest_compile_error')
74 delete('Xdir', 'rf')
Bram Moolenaarf4e8cdd2020-10-12 22:07:13 +020075enddef
76
Bram Moolenaarb55d6182021-06-08 22:01:53 +020077def Test_compile_error_in_called_function()
78 var lines =<< trim END
79 vim9script
80 var n: number
81 def Foo()
82 &hls = n
83 enddef
84 def Bar()
85 Foo()
86 enddef
87 silent! Foo()
88 Bar()
89 END
Bram Moolenaar62aec932022-01-29 21:45:34 +000090 v9.CheckScriptFailureList(lines, ['E1012:', 'E1191:'])
Bram Moolenaarb55d6182021-06-08 22:01:53 +020091enddef
92
Bram Moolenaar22f17a22021-06-21 20:48:58 +020093def Test_wrong_function_name()
94 var lines =<< trim END
95 vim9script
96 func _Foo()
97 echo 'foo'
98 endfunc
99 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000100 v9.CheckScriptFailure(lines, 'E128:')
Bram Moolenaar22f17a22021-06-21 20:48:58 +0200101
102 lines =<< trim END
103 vim9script
104 def _Foo()
105 echo 'foo'
106 enddef
107 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000108 v9.CheckScriptFailure(lines, 'E128:')
Bram Moolenaar22f17a22021-06-21 20:48:58 +0200109enddef
110
Bram Moolenaarf48b2fa2021-04-12 22:02:36 +0200111def Test_autoload_name_mismatch()
112 var dir = 'Xdir/autoload'
113 mkdir(dir, 'p')
114
115 var lines =<< trim END
116 vim9script
Bram Moolenaard8fe6d32022-01-30 18:40:44 +0000117 export def NoFunction()
Bram Moolenaarf48b2fa2021-04-12 22:02:36 +0200118 # comment
119 g:runtime = 'yes'
120 enddef
121 END
122 writefile(lines, dir .. '/script.vim')
123
124 var save_rtp = &rtp
125 exe 'set rtp=' .. getcwd() .. '/Xdir'
126 lines =<< trim END
127 call script#Function()
128 END
Bram Moolenaard8fe6d32022-01-30 18:40:44 +0000129 v9.CheckScriptFailure(lines, 'E117:', 1)
Bram Moolenaarf48b2fa2021-04-12 22:02:36 +0200130
131 &rtp = save_rtp
132 delete(dir, 'rf')
133enddef
134
Bram Moolenaarf0a40692021-06-11 22:05:47 +0200135def Test_autoload_names()
136 var dir = 'Xdir/autoload'
137 mkdir(dir, 'p')
138
139 var lines =<< trim END
140 func foobar#function()
141 return 'yes'
142 endfunc
143 let foobar#var = 'no'
144 END
145 writefile(lines, dir .. '/foobar.vim')
146
147 var save_rtp = &rtp
148 exe 'set rtp=' .. getcwd() .. '/Xdir'
149
150 lines =<< trim END
151 assert_equal('yes', foobar#function())
152 var Function = foobar#function
153 assert_equal('yes', Function())
154
155 assert_equal('no', foobar#var)
156 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000157 v9.CheckDefAndScriptSuccess(lines)
Bram Moolenaarf0a40692021-06-11 22:05:47 +0200158
159 &rtp = save_rtp
160 delete(dir, 'rf')
161enddef
162
Bram Moolenaar88c89c72021-08-14 14:01:05 +0200163def Test_autoload_error_in_script()
164 var dir = 'Xdir/autoload'
165 mkdir(dir, 'p')
166
167 var lines =<< trim END
168 func scripterror#function()
169 let g:called_function = 'yes'
170 endfunc
171 let 0 = 1
172 END
173 writefile(lines, dir .. '/scripterror.vim')
174
175 var save_rtp = &rtp
176 exe 'set rtp=' .. getcwd() .. '/Xdir'
177
178 g:called_function = 'no'
179 # The error in the autoload script cannot be checked with assert_fails(), use
180 # CheckDefSuccess() instead of CheckDefFailure()
181 try
Bram Moolenaar62aec932022-01-29 21:45:34 +0000182 v9.CheckDefSuccess(['scripterror#function()'])
Bram Moolenaar88c89c72021-08-14 14:01:05 +0200183 catch
184 assert_match('E121: Undefined variable: 0', v:exception)
185 endtry
186 assert_equal('no', g:called_function)
187
188 lines =<< trim END
189 func scriptcaught#function()
190 let g:called_function = 'yes'
191 endfunc
192 try
193 let 0 = 1
194 catch
195 let g:caught = v:exception
196 endtry
197 END
198 writefile(lines, dir .. '/scriptcaught.vim')
199
200 g:called_function = 'no'
Bram Moolenaar62aec932022-01-29 21:45:34 +0000201 v9.CheckDefSuccess(['scriptcaught#function()'])
Bram Moolenaar88c89c72021-08-14 14:01:05 +0200202 assert_match('E121: Undefined variable: 0', g:caught)
203 assert_equal('yes', g:called_function)
204
205 &rtp = save_rtp
206 delete(dir, 'rf')
207enddef
208
Bram Moolenaar62aec932022-01-29 21:45:34 +0000209def s:CallRecursive(n: number): number
Bram Moolenaar0ba48e82020-11-17 18:23:19 +0100210 return CallRecursive(n + 1)
211enddef
212
Bram Moolenaar62aec932022-01-29 21:45:34 +0000213def s:CallMapRecursive(l: list<number>): number
Bram Moolenaar2949cfd2020-12-31 21:28:47 +0100214 return map(l, (_, v) => CallMapRecursive([v]))[0]
Bram Moolenaar0ba48e82020-11-17 18:23:19 +0100215enddef
216
217def Test_funcdepth_error()
218 set maxfuncdepth=10
219
220 var caught = false
221 try
222 CallRecursive(1)
223 catch /E132:/
224 caught = true
225 endtry
226 assert_true(caught)
227
228 caught = false
229 try
230 CallMapRecursive([1])
231 catch /E132:/
232 caught = true
233 endtry
234 assert_true(caught)
235
236 set maxfuncdepth&
237enddef
238
Bram Moolenaar5178b1b2021-01-01 18:43:51 +0100239def Test_endfunc_enddef()
240 var lines =<< trim END
241 def Test()
242 echo 'test'
243 endfunc
244 enddef
245 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000246 v9.CheckScriptFailure(lines, 'E1151:', 3)
Bram Moolenaar5178b1b2021-01-01 18:43:51 +0100247
248 lines =<< trim END
249 def Test()
250 func Nested()
251 echo 'test'
252 enddef
253 enddef
254 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000255 v9.CheckScriptFailure(lines, 'E1152:', 4)
Bram Moolenaar49f1e9e2021-03-22 20:49:02 +0100256
257 lines =<< trim END
258 def Ok()
259 echo 'hello'
260 enddef | echo 'there'
261 def Bad()
262 echo 'hello'
263 enddef there
264 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000265 v9.CheckScriptFailure(lines, 'E1173: Text found after enddef: there', 6)
Bram Moolenaar5178b1b2021-01-01 18:43:51 +0100266enddef
267
Bram Moolenaarb8ba9b92021-01-01 18:54:34 +0100268def Test_missing_endfunc_enddef()
269 var lines =<< trim END
270 vim9script
271 def Test()
272 echo 'test'
273 endef
274 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000275 v9.CheckScriptFailure(lines, 'E1057:', 2)
Bram Moolenaarb8ba9b92021-01-01 18:54:34 +0100276
277 lines =<< trim END
278 vim9script
279 func Some()
280 echo 'test'
281 enfffunc
282 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000283 v9.CheckScriptFailure(lines, 'E126:', 2)
Bram Moolenaarb8ba9b92021-01-01 18:54:34 +0100284enddef
285
Bram Moolenaar4efd9942021-01-24 21:14:20 +0100286def Test_white_space_before_paren()
287 var lines =<< trim END
288 vim9script
289 def Test ()
290 echo 'test'
291 enddef
292 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000293 v9.CheckScriptFailure(lines, 'E1068:', 2)
Bram Moolenaar4efd9942021-01-24 21:14:20 +0100294
295 lines =<< trim END
296 vim9script
297 func Test ()
298 echo 'test'
299 endfunc
300 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000301 v9.CheckScriptFailure(lines, 'E1068:', 2)
Bram Moolenaar4efd9942021-01-24 21:14:20 +0100302
303 lines =<< trim END
304 def Test ()
305 echo 'test'
306 enddef
307 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000308 v9.CheckScriptFailure(lines, 'E1068:', 1)
Bram Moolenaar4efd9942021-01-24 21:14:20 +0100309
310 lines =<< trim END
311 func Test ()
312 echo 'test'
313 endfunc
314 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000315 v9.CheckScriptSuccess(lines)
Bram Moolenaar4efd9942021-01-24 21:14:20 +0100316enddef
317
Bram Moolenaar832ea892021-01-08 21:55:26 +0100318def Test_enddef_dict_key()
319 var d = {
320 enddef: 'x',
321 endfunc: 'y',
322 }
323 assert_equal({enddef: 'x', endfunc: 'y'}, d)
324enddef
325
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200326def ReturnString(): string
327 return 'string'
328enddef
329
330def ReturnNumber(): number
331 return 123
332enddef
333
334let g:notNumber = 'string'
335
336def ReturnGlobal(): number
337 return g:notNumber
338enddef
339
340def Test_return_something()
Bram Moolenaar62aec932022-01-29 21:45:34 +0000341 g:ReturnString()->assert_equal('string')
342 g:ReturnNumber()->assert_equal(123)
Bram Moolenaar848fadd2022-01-30 15:28:30 +0000343 assert_fails('g:ReturnGlobal()', 'E1012: Type mismatch; expected number but got string', '', 1, 'ReturnGlobal')
Bram Moolenaaref7aadb2022-01-18 18:46:07 +0000344
345 var lines =<< trim END
346 vim9script
347
348 def Msg()
349 echomsg 'in Msg()...'
350 enddef
351
352 def Func()
353 return Msg()
354 enddef
355 defcompile
356 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000357 v9.CheckScriptFailure(lines, 'E1096:')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200358enddef
359
Bram Moolenaare32e5162021-01-21 20:21:29 +0100360def Test_check_argument_type()
361 var lines =<< trim END
362 vim9script
363 def Val(a: number, b: number): number
364 return 0
365 enddef
366 def Func()
367 var x: any = true
368 Val(0, x)
369 enddef
370 disass Func
371 Func()
372 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000373 v9.CheckScriptFailure(lines, 'E1013: Argument 2: type mismatch, expected number but got bool', 2)
Bram Moolenaare32e5162021-01-21 20:21:29 +0100374enddef
375
Bram Moolenaarefd88552020-06-18 20:50:10 +0200376def Test_missing_return()
Bram Moolenaar62aec932022-01-29 21:45:34 +0000377 v9.CheckDefFailure(['def Missing(): number',
Bram Moolenaarefd88552020-06-18 20:50:10 +0200378 ' if g:cond',
379 ' echo "no return"',
380 ' else',
381 ' return 0',
382 ' endif'
383 'enddef'], 'E1027:')
Bram Moolenaar62aec932022-01-29 21:45:34 +0000384 v9.CheckDefFailure(['def Missing(): number',
Bram Moolenaarefd88552020-06-18 20:50:10 +0200385 ' if g:cond',
386 ' return 1',
387 ' else',
388 ' echo "no return"',
389 ' endif'
390 'enddef'], 'E1027:')
Bram Moolenaar62aec932022-01-29 21:45:34 +0000391 v9.CheckDefFailure(['def Missing(): number',
Bram Moolenaarefd88552020-06-18 20:50:10 +0200392 ' if g:cond',
393 ' return 1',
394 ' else',
395 ' return 2',
396 ' endif'
397 ' return 3'
398 'enddef'], 'E1095:')
399enddef
400
Bram Moolenaar403dc312020-10-17 19:29:51 +0200401def Test_return_bool()
402 var lines =<< trim END
403 vim9script
404 def MenuFilter(id: number, key: string): bool
405 return popup_filter_menu(id, key)
406 enddef
407 def YesnoFilter(id: number, key: string): bool
408 return popup_filter_yesno(id, key)
409 enddef
410 defcompile
411 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000412 v9.CheckScriptSuccess(lines)
Bram Moolenaar403dc312020-10-17 19:29:51 +0200413enddef
414
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200415let s:nothing = 0
416def ReturnNothing()
417 s:nothing = 1
418 if true
419 return
420 endif
421 s:nothing = 2
422enddef
423
424def Test_return_nothing()
Bram Moolenaar62aec932022-01-29 21:45:34 +0000425 g:ReturnNothing()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +0200426 s:nothing->assert_equal(1)
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200427enddef
428
Bram Moolenaar648ea762021-01-15 19:04:32 +0100429def Test_return_invalid()
430 var lines =<< trim END
431 vim9script
432 def Func(): invalid
433 return xxx
434 enddef
435 defcompile
436 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000437 v9.CheckScriptFailure(lines, 'E1010:', 2)
Bram Moolenaar31842cd2021-02-12 22:10:21 +0100438
439 lines =<< trim END
440 vim9script
441 def Test(Fun: func(number): number): list<number>
442 return map([1, 2, 3], (_, i) => Fun(i))
443 enddef
444 defcompile
445 def Inc(nr: number): nr
446 return nr + 2
447 enddef
448 echo Test(Inc)
449 END
450 # doing this twice was leaking memory
Bram Moolenaar62aec932022-01-29 21:45:34 +0000451 v9.CheckScriptFailure(lines, 'E1010:')
452 v9.CheckScriptFailure(lines, 'E1010:')
Bram Moolenaar648ea762021-01-15 19:04:32 +0100453enddef
454
Bram Moolenaarefc084e2021-09-09 22:30:52 +0200455def Test_return_list_any()
Bram Moolenaar114dbda2022-01-03 12:28:03 +0000456 # This used to fail but now the actual list type is checked, and since it has
457 # an item of type string it can be used as list<string>.
Bram Moolenaarefc084e2021-09-09 22:30:52 +0200458 var lines =<< trim END
459 vim9script
460 def Func(): list<string>
461 var l: list<any>
462 l->add('string')
463 return l
464 enddef
465 echo Func()
466 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000467 v9.CheckScriptSuccess(lines)
Bram Moolenaar114dbda2022-01-03 12:28:03 +0000468
Bram Moolenaarefc084e2021-09-09 22:30:52 +0200469 lines =<< trim END
470 vim9script
471 def Func(): list<string>
472 var l: list<any>
473 l += ['string']
474 return l
475 enddef
476 echo Func()
477 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000478 v9.CheckScriptSuccess(lines)
Bram Moolenaarefc084e2021-09-09 22:30:52 +0200479enddef
480
Bram Moolenaar62aec932022-01-29 21:45:34 +0000481func s:Increment()
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200482 let g:counter += 1
483endfunc
484
485def Test_call_ufunc_count()
486 g:counter = 1
487 Increment()
488 Increment()
489 Increment()
Bram Moolenaarf5be8cd2020-07-17 20:36:00 +0200490 # works with and without :call
Bram Moolenaarc0c71e92020-09-11 19:09:48 +0200491 g:counter->assert_equal(4)
492 eval g:counter->assert_equal(4)
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200493 unlet g:counter
494enddef
495
Bram Moolenaar62aec932022-01-29 21:45:34 +0000496def s:MyVarargs(arg: string, ...rest: list<string>): string
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +0200497 var res = arg
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200498 for s in rest
499 res ..= ',' .. s
500 endfor
501 return res
502enddef
503
504def Test_call_varargs()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +0200505 MyVarargs('one')->assert_equal('one')
506 MyVarargs('one', 'two')->assert_equal('one,two')
507 MyVarargs('one', 'two', 'three')->assert_equal('one,two,three')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200508enddef
509
Bram Moolenaar01dd6c32021-09-05 16:36:23 +0200510def Test_call_white_space()
Bram Moolenaar62aec932022-01-29 21:45:34 +0000511 v9.CheckDefAndScriptFailure(["call Test ('text')"], ['E476:', 'E1068:'])
Bram Moolenaar01dd6c32021-09-05 16:36:23 +0200512enddef
513
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200514def MyDefaultArgs(name = 'string'): string
515 return name
516enddef
517
Bram Moolenaar62aec932022-01-29 21:45:34 +0000518def s:MyDefaultSecond(name: string, second: bool = true): string
Bram Moolenaare30f64b2020-07-15 19:48:20 +0200519 return second ? name : 'none'
520enddef
521
Bram Moolenaar38a3bfa2021-03-29 22:14:55 +0200522
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200523def Test_call_default_args()
Bram Moolenaar62aec932022-01-29 21:45:34 +0000524 g:MyDefaultArgs()->assert_equal('string')
525 g:MyDefaultArgs(v:none)->assert_equal('string')
526 g:MyDefaultArgs('one')->assert_equal('one')
527 assert_fails('g:MyDefaultArgs("one", "two")', 'E118:', '', 4, 'Test_call_default_args')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200528
Bram Moolenaarc0c71e92020-09-11 19:09:48 +0200529 MyDefaultSecond('test')->assert_equal('test')
530 MyDefaultSecond('test', true)->assert_equal('test')
531 MyDefaultSecond('test', false)->assert_equal('none')
Bram Moolenaare30f64b2020-07-15 19:48:20 +0200532
Bram Moolenaar38a3bfa2021-03-29 22:14:55 +0200533 var lines =<< trim END
534 def MyDefaultThird(name: string, aa = 'aa', bb = 'bb'): string
535 return name .. aa .. bb
536 enddef
537
538 MyDefaultThird('->')->assert_equal('->aabb')
539 MyDefaultThird('->', v:none)->assert_equal('->aabb')
540 MyDefaultThird('->', 'xx')->assert_equal('->xxbb')
541 MyDefaultThird('->', v:none, v:none)->assert_equal('->aabb')
542 MyDefaultThird('->', 'xx', v:none)->assert_equal('->xxbb')
543 MyDefaultThird('->', v:none, 'yy')->assert_equal('->aayy')
544 MyDefaultThird('->', 'xx', 'yy')->assert_equal('->xxyy')
Bram Moolenaare28d9b32021-07-03 18:56:53 +0200545
546 def DefArg(mandatory: any, optional = mandatory): string
547 return mandatory .. optional
548 enddef
549 DefArg(1234)->assert_equal('12341234')
550 DefArg("ok")->assert_equal('okok')
Bram Moolenaar38a3bfa2021-03-29 22:14:55 +0200551 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000552 v9.CheckDefAndScriptSuccess(lines)
Bram Moolenaar38a3bfa2021-03-29 22:14:55 +0200553
Bram Moolenaar62aec932022-01-29 21:45:34 +0000554 v9.CheckScriptFailure(['def Func(arg: number = asdf)', 'enddef', 'defcompile'], 'E1001:')
Bram Moolenaar2d870f82020-12-05 13:41:01 +0100555 delfunc g:Func
Bram Moolenaar62aec932022-01-29 21:45:34 +0000556 v9.CheckScriptFailure(['def Func(arg: number = "text")', 'enddef', 'defcompile'], 'E1013: Argument 1: type mismatch, expected number but got string')
Bram Moolenaar2d870f82020-12-05 13:41:01 +0100557 delfunc g:Func
Bram Moolenaar62aec932022-01-29 21:45:34 +0000558 v9.CheckDefFailure(['def Func(x: number = )', 'enddef'], 'E15:')
Bram Moolenaar12bce952021-03-11 20:04:04 +0100559
Bram Moolenaar38a3bfa2021-03-29 22:14:55 +0200560 lines =<< trim END
Bram Moolenaar12bce952021-03-11 20:04:04 +0100561 vim9script
562 def Func(a = b == 0 ? 1 : 2, b = 0)
563 enddef
564 defcompile
565 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000566 v9.CheckScriptFailure(lines, 'E1001: Variable not found: b')
Bram Moolenaar59618fe2021-12-21 12:32:17 +0000567
Bram Moolenaarfa46ead2021-12-22 13:18:39 +0000568 # using script variable requires matching type or type cast when executed
Bram Moolenaar59618fe2021-12-21 12:32:17 +0000569 lines =<< trim END
570 vim9script
571 var a: any
572 def Func(arg: string = a)
573 echo arg
574 enddef
575 defcompile
576 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000577 v9.CheckScriptSuccess(lines + ['a = "text"', 'Func()'])
578 v9.CheckScriptFailure(lines + ['a = 123', 'Func()'], 'E1013: Argument 1: type mismatch, expected string but got number')
Bram Moolenaar59618fe2021-12-21 12:32:17 +0000579
580 # using global variable does not require type cast
581 lines =<< trim END
582 vim9script
583 def Func(arg: string = g:str)
584 echo arg
585 enddef
586 g:str = 'works'
587 Func()
588 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000589 v9.CheckScriptSuccess(lines)
Bram Moolenaar04b12692020-05-04 23:24:44 +0200590enddef
591
Bram Moolenaar62aec932022-01-29 21:45:34 +0000592def s:FuncWithComment( # comment
Bram Moolenaarcef12702021-01-04 14:09:43 +0100593 a: number, #comment
594 b: bool, # comment
595 c: string) #comment
596 assert_equal(4, a)
597 assert_equal(true, b)
598 assert_equal('yes', c)
599enddef
600
601def Test_func_with_comments()
602 FuncWithComment(4, true, 'yes')
603
604 var lines =<< trim END
605 def Func(# comment
606 arg: string)
607 enddef
608 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000609 v9.CheckScriptFailure(lines, 'E125:', 1)
Bram Moolenaarcef12702021-01-04 14:09:43 +0100610
611 lines =<< trim END
612 def Func(
613 arg: string# comment
614 )
615 enddef
616 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000617 v9.CheckScriptFailure(lines, 'E475:', 2)
Bram Moolenaarcef12702021-01-04 14:09:43 +0100618
619 lines =<< trim END
620 def Func(
621 arg: string
622 )# comment
623 enddef
624 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000625 v9.CheckScriptFailure(lines, 'E488:', 3)
Bram Moolenaarcef12702021-01-04 14:09:43 +0100626enddef
627
Bram Moolenaar04b12692020-05-04 23:24:44 +0200628def Test_nested_function()
Bram Moolenaar38453522021-11-28 22:00:12 +0000629 def NestedDef(arg: string): string
Bram Moolenaar04b12692020-05-04 23:24:44 +0200630 return 'nested ' .. arg
631 enddef
Bram Moolenaar38453522021-11-28 22:00:12 +0000632 NestedDef(':def')->assert_equal('nested :def')
633
634 func NestedFunc(arg)
635 return 'nested ' .. a:arg
636 endfunc
637 NestedFunc(':func')->assert_equal('nested :func')
Bram Moolenaar04b12692020-05-04 23:24:44 +0200638
Bram Moolenaar62aec932022-01-29 21:45:34 +0000639 v9.CheckDefFailure(['def Nested()', 'enddef', 'Nested(66)'], 'E118:')
640 v9.CheckDefFailure(['def Nested(arg: string)', 'enddef', 'Nested()'], 'E119:')
Bram Moolenaar0e65d3d2020-05-05 17:53:16 +0200641
Bram Moolenaar62aec932022-01-29 21:45:34 +0000642 v9.CheckDefFailure(['def s:Nested()', 'enddef'], 'E1075:')
643 v9.CheckDefFailure(['def b:Nested()', 'enddef'], 'E1075:')
Bram Moolenaar8b848ca2020-09-10 22:28:01 +0200644
Bram Moolenaar54021752020-12-06 18:50:36 +0100645 var lines =<< trim END
646 def Outer()
647 def Inner()
648 # comment
649 enddef
650 def Inner()
651 enddef
652 enddef
653 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000654 v9.CheckDefFailure(lines, 'E1073:')
Bram Moolenaar54021752020-12-06 18:50:36 +0100655
656 lines =<< trim END
657 def Outer()
658 def Inner()
659 # comment
660 enddef
661 def! Inner()
662 enddef
663 enddef
664 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000665 v9.CheckDefFailure(lines, 'E1117:')
Bram Moolenaar54021752020-12-06 18:50:36 +0100666
Bram Moolenaardb8e5c22021-12-25 19:58:22 +0000667 lines =<< trim END
668 vim9script
669 def Outer()
670 def Inner()
671 g:result = 'ok'
672 enddef
673 Inner()
674 enddef
675 Outer()
676 Inner()
677 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000678 v9.CheckScriptFailure(lines, 'E117: Unknown function: Inner')
Bram Moolenaardb8e5c22021-12-25 19:58:22 +0000679 assert_equal('ok', g:result)
680 unlet g:result
681
Bram Moolenaarf681cfb2022-02-07 20:30:57 +0000682 lines =<< trim END
683 vim9script
684 def Outer()
685 def _Inner()
686 echo 'bad'
687 enddef
688 Inner()
689 enddef
690 defcompile
691 END
692 v9.CheckScriptFailure(lines, 'E128:')
693
694 lines =<< trim END
695 vim9script
696 def Outer()
697 def g:inner()
698 echo 'bad'
699 enddef
700 Inner()
701 enddef
702 defcompile
703 END
704 v9.CheckScriptFailure(lines, 'E128:')
705
Bram Moolenaar54021752020-12-06 18:50:36 +0100706 # nested function inside conditional
Bram Moolenaar54021752020-12-06 18:50:36 +0100707 lines =<< trim END
708 vim9script
709 var thecount = 0
710 if true
711 def Test(): number
712 def TheFunc(): number
713 thecount += 1
714 return thecount
715 enddef
716 return TheFunc()
717 enddef
718 endif
719 defcompile
720 assert_equal(1, Test())
721 assert_equal(2, Test())
722 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000723 v9.CheckScriptSuccess(lines)
Bram Moolenaar8863bda2021-03-17 18:42:08 +0100724
725 # also works when "thecount" is inside the "if" block
726 lines =<< trim END
727 vim9script
728 if true
729 var thecount = 0
730 def Test(): number
731 def TheFunc(): number
732 thecount += 1
733 return thecount
734 enddef
735 return TheFunc()
736 enddef
737 endif
738 defcompile
739 assert_equal(1, Test())
740 assert_equal(2, Test())
741 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000742 v9.CheckScriptSuccess(lines)
Bram Moolenaar4bba16d2021-08-15 19:28:05 +0200743
744 lines =<< trim END
745 vim9script
746 def Outer()
747 def Inner()
748 echo 'hello'
749 enddef burp
750 enddef
751 defcompile
752 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000753 v9.CheckScriptFailure(lines, 'E1173: Text found after enddef: burp', 3)
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200754enddef
755
Bram Moolenaaradc8e442020-12-31 18:28:18 +0100756def Test_not_nested_function()
757 echo printf('%d',
758 function('len')('xxx'))
759enddef
760
Bram Moolenaaraf8edbb2020-08-01 00:03:09 +0200761func Test_call_default_args_from_func()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +0200762 call MyDefaultArgs()->assert_equal('string')
763 call MyDefaultArgs('one')->assert_equal('one')
Bram Moolenaar9bd5d872020-09-06 21:47:48 +0200764 call assert_fails('call MyDefaultArgs("one", "two")', 'E118:', '', 3, 'Test_call_default_args_from_func')
Bram Moolenaaraf8edbb2020-08-01 00:03:09 +0200765endfunc
766
Bram Moolenaar38ddf332020-07-31 22:05:04 +0200767def Test_nested_global_function()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +0200768 var lines =<< trim END
Bram Moolenaar38ddf332020-07-31 22:05:04 +0200769 vim9script
770 def Outer()
771 def g:Inner(): string
772 return 'inner'
773 enddef
774 enddef
Bram Moolenaaraf8edbb2020-08-01 00:03:09 +0200775 defcompile
776 Outer()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +0200777 g:Inner()->assert_equal('inner')
Bram Moolenaaraf8edbb2020-08-01 00:03:09 +0200778 delfunc g:Inner
779 Outer()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +0200780 g:Inner()->assert_equal('inner')
Bram Moolenaaraf8edbb2020-08-01 00:03:09 +0200781 delfunc g:Inner
782 Outer()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +0200783 g:Inner()->assert_equal('inner')
Bram Moolenaaraf8edbb2020-08-01 00:03:09 +0200784 delfunc g:Inner
Bram Moolenaar38ddf332020-07-31 22:05:04 +0200785 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000786 v9.CheckScriptSuccess(lines)
Bram Moolenaar2c79e9d2020-08-01 18:57:52 +0200787
788 lines =<< trim END
789 vim9script
790 def Outer()
Bram Moolenaar38453522021-11-28 22:00:12 +0000791 func g:Inner()
792 return 'inner'
793 endfunc
794 enddef
795 defcompile
796 Outer()
797 g:Inner()->assert_equal('inner')
798 delfunc g:Inner
799 Outer()
800 g:Inner()->assert_equal('inner')
801 delfunc g:Inner
802 Outer()
803 g:Inner()->assert_equal('inner')
804 delfunc g:Inner
805 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000806 v9.CheckScriptSuccess(lines)
Bram Moolenaar38453522021-11-28 22:00:12 +0000807
808 lines =<< trim END
809 vim9script
810 def Outer()
Bram Moolenaar2c79e9d2020-08-01 18:57:52 +0200811 def g:Inner(): string
812 return 'inner'
813 enddef
814 enddef
815 defcompile
816 Outer()
817 Outer()
818 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000819 v9.CheckScriptFailure(lines, "E122:")
Bram Moolenaarcd45ed02020-12-22 17:35:54 +0100820 delfunc g:Inner
Bram Moolenaarad486a02020-08-01 23:22:18 +0200821
822 lines =<< trim END
823 vim9script
Bram Moolenaar58a52f22020-12-22 18:56:55 +0100824 def Outer()
825 def g:Inner()
Bram Moolenaar2949cfd2020-12-31 21:28:47 +0100826 echo map([1, 2, 3], (_, v) => v + 1)
Bram Moolenaar58a52f22020-12-22 18:56:55 +0100827 enddef
828 g:Inner()
829 enddef
830 Outer()
831 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000832 v9.CheckScriptSuccess(lines)
Bram Moolenaar58a52f22020-12-22 18:56:55 +0100833 delfunc g:Inner
834
835 lines =<< trim END
836 vim9script
Bram Moolenaarad486a02020-08-01 23:22:18 +0200837 def Func()
838 echo 'script'
839 enddef
840 def Outer()
841 def Func()
842 echo 'inner'
843 enddef
844 enddef
845 defcompile
846 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000847 v9.CheckScriptFailure(lines, "E1073:", 1)
Bram Moolenaard604d782021-11-20 21:46:20 +0000848
849 lines =<< trim END
850 vim9script
851 def Func()
852 echo 'script'
853 enddef
854 def Func()
855 echo 'script'
856 enddef
857 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000858 v9.CheckScriptFailure(lines, "E1073:", 5)
Bram Moolenaar38ddf332020-07-31 22:05:04 +0200859enddef
860
Bram Moolenaar6abdcf82020-11-22 18:15:44 +0100861def DefListAll()
862 def
863enddef
864
865def DefListOne()
866 def DefListOne
867enddef
868
869def DefListMatches()
870 def /DefList
871enddef
872
873def Test_nested_def_list()
874 var funcs = split(execute('call DefListAll()'), "\n")
875 assert_true(len(funcs) > 10)
876 assert_true(funcs->index('def DefListAll()') >= 0)
877
878 funcs = split(execute('call DefListOne()'), "\n")
879 assert_equal([' def DefListOne()', '1 def DefListOne', ' enddef'], funcs)
880
881 funcs = split(execute('call DefListMatches()'), "\n")
882 assert_true(len(funcs) >= 3)
883 assert_true(funcs->index('def DefListAll()') >= 0)
884 assert_true(funcs->index('def DefListOne()') >= 0)
885 assert_true(funcs->index('def DefListMatches()') >= 0)
Bram Moolenaar54021752020-12-06 18:50:36 +0100886
887 var lines =<< trim END
888 vim9script
889 def Func()
890 def +Func+
891 enddef
892 defcompile
893 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000894 v9.CheckScriptFailure(lines, 'E476:', 1)
Bram Moolenaar6abdcf82020-11-22 18:15:44 +0100895enddef
896
Bram Moolenaar333894b2020-08-01 18:53:07 +0200897def Test_global_local_function()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +0200898 var lines =<< trim END
Bram Moolenaar333894b2020-08-01 18:53:07 +0200899 vim9script
900 def g:Func(): string
901 return 'global'
902 enddef
903 def Func(): string
904 return 'local'
905 enddef
Bram Moolenaarc0c71e92020-09-11 19:09:48 +0200906 g:Func()->assert_equal('global')
907 Func()->assert_equal('local')
Bram Moolenaar2d870f82020-12-05 13:41:01 +0100908 delfunc g:Func
Bram Moolenaar333894b2020-08-01 18:53:07 +0200909 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000910 v9.CheckScriptSuccess(lines)
Bram Moolenaar035d6e92020-08-11 22:30:42 +0200911
912 lines =<< trim END
913 vim9script
914 def g:Funcy()
915 echo 'funcy'
916 enddef
917 s:Funcy()
918 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000919 v9.CheckScriptFailure(lines, 'E117:')
Bram Moolenaar333894b2020-08-01 18:53:07 +0200920enddef
921
Bram Moolenaar0f769812020-09-12 18:32:34 +0200922def Test_local_function_shadows_global()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +0200923 var lines =<< trim END
Bram Moolenaar0f769812020-09-12 18:32:34 +0200924 vim9script
925 def g:Gfunc(): string
926 return 'global'
927 enddef
928 def AnotherFunc(): number
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +0200929 var Gfunc = function('len')
Bram Moolenaar0f769812020-09-12 18:32:34 +0200930 return Gfunc('testing')
931 enddef
932 g:Gfunc()->assert_equal('global')
933 AnotherFunc()->assert_equal(7)
934 delfunc g:Gfunc
935 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000936 v9.CheckScriptSuccess(lines)
Bram Moolenaar0f769812020-09-12 18:32:34 +0200937
938 lines =<< trim END
939 vim9script
940 def g:Func(): string
941 return 'global'
942 enddef
943 def AnotherFunc()
944 g:Func = function('len')
945 enddef
946 AnotherFunc()
947 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000948 v9.CheckScriptFailure(lines, 'E705:')
Bram Moolenaar0f769812020-09-12 18:32:34 +0200949 delfunc g:Func
Bram Moolenaar0865b152021-04-05 15:38:51 +0200950
Bram Moolenaar62aec932022-01-29 21:45:34 +0000951 # global function is not found with g: prefix
Bram Moolenaar0865b152021-04-05 15:38:51 +0200952 lines =<< trim END
953 vim9script
954 def g:Func(): string
955 return 'global'
956 enddef
957 def AnotherFunc(): string
958 return Func()
959 enddef
960 assert_equal('global', AnotherFunc())
Bram Moolenaar0865b152021-04-05 15:38:51 +0200961 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000962 v9.CheckScriptFailure(lines, 'E117:')
963 delfunc g:Func
Bram Moolenaar0865b152021-04-05 15:38:51 +0200964
965 lines =<< trim END
966 vim9script
967 def g:Func(): string
968 return 'global'
969 enddef
Bram Moolenaar848fadd2022-01-30 15:28:30 +0000970 assert_equal('global', g:Func())
Bram Moolenaar0865b152021-04-05 15:38:51 +0200971 delfunc g:Func
972 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000973 v9.CheckScriptSuccess(lines)
Bram Moolenaar58493cf2022-01-06 12:23:30 +0000974
975 # This does not shadow "i" which is visible only inside the for loop
976 lines =<< trim END
977 vim9script
978
979 def Foo(i: number)
980 echo i
981 enddef
982
983 for i in range(3)
984 # Foo() is compiled here
985 Foo(i)
986 endfor
987 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000988 v9.CheckScriptSuccess(lines)
Bram Moolenaar0f769812020-09-12 18:32:34 +0200989enddef
990
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200991func TakesOneArg(arg)
992 echo a:arg
993endfunc
994
995def Test_call_wrong_args()
Bram Moolenaar62aec932022-01-29 21:45:34 +0000996 v9.CheckDefFailure(['g:TakesOneArg()'], 'E119:')
997 v9.CheckDefFailure(['g:TakesOneArg(11, 22)'], 'E118:')
998 v9.CheckDefFailure(['bufnr(xxx)'], 'E1001:')
999 v9.CheckScriptFailure(['def Func(Ref: func(s: string))'], 'E475:')
Bram Moolenaaree8580e2020-08-28 17:19:07 +02001000
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001001 var lines =<< trim END
Bram Moolenaaree8580e2020-08-28 17:19:07 +02001002 vim9script
1003 def Func(s: string)
1004 echo s
1005 enddef
1006 Func([])
1007 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001008 v9.CheckScriptFailure(lines, 'E1013: Argument 1: type mismatch, expected string but got list<unknown>', 5)
Bram Moolenaarb185a402020-09-18 22:42:00 +02001009
Bram Moolenaar9a015112021-12-31 14:06:45 +00001010 # argument name declared earlier is found when declaring a function
Bram Moolenaarb185a402020-09-18 22:42:00 +02001011 lines =<< trim END
1012 vim9script
Bram Moolenaarb4893b82021-02-21 22:20:24 +01001013 var name = 'piet'
1014 def FuncOne(name: string)
1015 echo nr
1016 enddef
1017 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001018 v9.CheckScriptFailure(lines, 'E1168:')
Bram Moolenaarb4893b82021-02-21 22:20:24 +01001019
Bram Moolenaar9a015112021-12-31 14:06:45 +00001020 # argument name declared later is only found when compiling
1021 lines =<< trim END
1022 vim9script
1023 def FuncOne(name: string)
1024 echo nr
1025 enddef
1026 var name = 'piet'
1027 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001028 v9.CheckScriptSuccess(lines)
1029 v9.CheckScriptFailure(lines + ['defcompile'], 'E1168:')
Bram Moolenaar9a015112021-12-31 14:06:45 +00001030
Bram Moolenaarb4893b82021-02-21 22:20:24 +01001031 lines =<< trim END
1032 vim9script
Bram Moolenaarb185a402020-09-18 22:42:00 +02001033 def FuncOne(nr: number)
1034 echo nr
1035 enddef
1036 def FuncTwo()
1037 FuncOne()
1038 enddef
1039 defcompile
1040 END
1041 writefile(lines, 'Xscript')
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001042 var didCatch = false
Bram Moolenaarb185a402020-09-18 22:42:00 +02001043 try
1044 source Xscript
1045 catch
1046 assert_match('E119: Not enough arguments for function: <SNR>\d\+_FuncOne', v:exception)
1047 assert_match('Xscript\[8\]..function <SNR>\d\+_FuncTwo, line 1', v:throwpoint)
1048 didCatch = true
1049 endtry
1050 assert_true(didCatch)
1051
1052 lines =<< trim END
1053 vim9script
1054 def FuncOne(nr: number)
1055 echo nr
1056 enddef
1057 def FuncTwo()
1058 FuncOne(1, 2)
1059 enddef
1060 defcompile
1061 END
1062 writefile(lines, 'Xscript')
1063 didCatch = false
1064 try
1065 source Xscript
1066 catch
1067 assert_match('E118: Too many arguments for function: <SNR>\d\+_FuncOne', v:exception)
1068 assert_match('Xscript\[8\]..function <SNR>\d\+_FuncTwo, line 1', v:throwpoint)
1069 didCatch = true
1070 endtry
1071 assert_true(didCatch)
1072
1073 delete('Xscript')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001074enddef
1075
Bram Moolenaar50824712020-12-20 21:10:17 +01001076def Test_call_funcref_wrong_args()
1077 var head =<< trim END
1078 vim9script
1079 def Func3(a1: string, a2: number, a3: list<number>)
1080 echo a1 .. a2 .. a3[0]
1081 enddef
1082 def Testme()
1083 var funcMap: dict<func> = {func: Func3}
1084 END
1085 var tail =<< trim END
1086 enddef
1087 Testme()
1088 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001089 v9.CheckScriptSuccess(head + ["funcMap['func']('str', 123, [1, 2, 3])"] + tail)
Bram Moolenaar50824712020-12-20 21:10:17 +01001090
Bram Moolenaar62aec932022-01-29 21:45:34 +00001091 v9.CheckScriptFailure(head + ["funcMap['func']('str', 123)"] + tail, 'E119:')
1092 v9.CheckScriptFailure(head + ["funcMap['func']('str', 123, [1], 4)"] + tail, 'E118:')
Bram Moolenaar32b3f822021-01-06 21:59:39 +01001093
1094 var lines =<< trim END
1095 vim9script
1096 var Ref: func(number): any
1097 Ref = (j) => !j
1098 echo Ref(false)
1099 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001100 v9.CheckScriptFailure(lines, 'E1013: Argument 1: type mismatch, expected number but got bool', 4)
Bram Moolenaar32b3f822021-01-06 21:59:39 +01001101
1102 lines =<< trim END
1103 vim9script
1104 var Ref: func(number): any
1105 Ref = (j) => !j
1106 call Ref(false)
1107 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001108 v9.CheckScriptFailure(lines, 'E1013: Argument 1: type mismatch, expected number but got bool', 4)
Bram Moolenaar50824712020-12-20 21:10:17 +01001109enddef
1110
Bram Moolenaarb4d16cb2020-11-05 18:45:46 +01001111def Test_call_lambda_args()
Bram Moolenaar2a389082021-04-09 20:24:31 +02001112 var lines =<< trim END
1113 var Callback = (..._) => 'anything'
1114 assert_equal('anything', Callback())
1115 assert_equal('anything', Callback(1))
1116 assert_equal('anything', Callback('a', 2))
Bram Moolenaar1088b692021-04-09 22:12:44 +02001117
1118 assert_equal('xyz', ((a: string): string => a)('xyz'))
Bram Moolenaar2a389082021-04-09 20:24:31 +02001119 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001120 v9.CheckDefAndScriptSuccess(lines)
Bram Moolenaar2a389082021-04-09 20:24:31 +02001121
Bram Moolenaar62aec932022-01-29 21:45:34 +00001122 v9.CheckDefFailure(['echo ((i) => 0)()'],
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01001123 'E119: Not enough arguments for function: ((i) => 0)()')
Bram Moolenaarb4d16cb2020-11-05 18:45:46 +01001124
Bram Moolenaar2a389082021-04-09 20:24:31 +02001125 lines =<< trim END
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01001126 var Ref = (x: number, y: number) => x + y
Bram Moolenaarb4d16cb2020-11-05 18:45:46 +01001127 echo Ref(1, 'x')
1128 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001129 v9.CheckDefFailure(lines, 'E1013: Argument 2: type mismatch, expected number but got string')
Bram Moolenaare68b02a2021-01-03 13:09:51 +01001130
1131 lines =<< trim END
1132 var Ref: func(job, string, number)
1133 Ref = (x, y) => 0
1134 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001135 v9.CheckDefAndScriptFailure(lines, 'E1012:')
Bram Moolenaare68b02a2021-01-03 13:09:51 +01001136
1137 lines =<< trim END
1138 var Ref: func(job, string)
1139 Ref = (x, y, z) => 0
1140 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001141 v9.CheckDefAndScriptFailure(lines, 'E1012:')
Bram Moolenaar057e84a2021-02-28 16:55:11 +01001142
1143 lines =<< trim END
1144 var one = 1
1145 var l = [1, 2, 3]
1146 echo map(l, (one) => one)
1147 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001148 v9.CheckDefFailure(lines, 'E1167:')
1149 v9.CheckScriptFailure(['vim9script'] + lines, 'E1168:')
Bram Moolenaar057e84a2021-02-28 16:55:11 +01001150
1151 lines =<< trim END
Bram Moolenaar14ded112021-06-26 19:25:49 +02001152 var Ref: func(any, ?any): bool
1153 Ref = (_, y = 1) => false
1154 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001155 v9.CheckDefAndScriptFailure(lines, 'E1172:')
Bram Moolenaar14ded112021-06-26 19:25:49 +02001156
1157 lines =<< trim END
Bram Moolenaar015cf102021-06-26 21:52:02 +02001158 var a = 0
1159 var b = (a == 0 ? 1 : 2)
1160 assert_equal(1, b)
Bram Moolenaar98f9a5f2021-06-26 22:22:38 +02001161 var txt = 'a'
1162 b = (txt =~ 'x' ? 1 : 2)
1163 assert_equal(2, b)
Bram Moolenaar015cf102021-06-26 21:52:02 +02001164 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001165 v9.CheckDefAndScriptSuccess(lines)
Bram Moolenaar015cf102021-06-26 21:52:02 +02001166
1167 lines =<< trim END
Bram Moolenaar057e84a2021-02-28 16:55:11 +01001168 def ShadowLocal()
1169 var one = 1
1170 var l = [1, 2, 3]
1171 echo map(l, (one) => one)
1172 enddef
1173 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001174 v9.CheckDefFailure(lines, 'E1167:')
Bram Moolenaar057e84a2021-02-28 16:55:11 +01001175
1176 lines =<< trim END
1177 def Shadowarg(one: number)
1178 var l = [1, 2, 3]
1179 echo map(l, (one) => one)
1180 enddef
1181 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001182 v9.CheckDefFailure(lines, 'E1167:')
Bram Moolenaar767034c2021-04-09 17:24:52 +02001183
1184 lines =<< trim END
1185 echo ((a) => a)('aa', 'bb')
1186 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001187 v9.CheckDefAndScriptFailure(lines, 'E118:', 1)
Bram Moolenaarc4c56422021-07-21 20:38:46 +02001188
1189 lines =<< trim END
1190 echo 'aa'->((a) => a)('bb')
1191 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001192 v9.CheckDefFailure(lines, 'E118: Too many arguments for function: ->((a) => a)(''bb'')', 1)
1193 v9.CheckScriptFailure(['vim9script'] + lines, 'E118: Too many arguments for function: <lambda>', 2)
Bram Moolenaarb4d16cb2020-11-05 18:45:46 +01001194enddef
1195
Bram Moolenaara755fdb2021-11-20 21:35:41 +00001196def Test_lambda_line_nr()
1197 var lines =<< trim END
1198 vim9script
1199 # comment
1200 # comment
1201 var id = timer_start(1'000, (_) => 0)
1202 var out = execute('verbose ' .. timer_info(id)[0].callback
1203 ->string()
1204 ->substitute("('\\|')", ' ', 'g'))
1205 assert_match('Last set from .* line 4', out)
1206 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001207 v9.CheckScriptSuccess(lines)
Bram Moolenaara755fdb2021-11-20 21:35:41 +00001208enddef
1209
Bram Moolenaar5f91e742021-03-17 21:29:29 +01001210def FilterWithCond(x: string, Cond: func(string): bool): bool
1211 return Cond(x)
1212enddef
1213
Bram Moolenaar0346b792021-01-31 22:18:29 +01001214def Test_lambda_return_type()
1215 var lines =<< trim END
1216 var Ref = (): => 123
1217 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001218 v9.CheckDefAndScriptFailure(lines, 'E1157:', 1)
Bram Moolenaar5f91e742021-03-17 21:29:29 +01001219
Yegappan Lakshmanan611728f2021-05-24 15:15:47 +02001220 # no space before the return type
1221 lines =<< trim END
1222 var Ref = (x):number => x + 1
1223 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001224 v9.CheckDefAndScriptFailure(lines, 'E1069:', 1)
Yegappan Lakshmanan611728f2021-05-24 15:15:47 +02001225
Bram Moolenaar5f91e742021-03-17 21:29:29 +01001226 # this works
1227 for x in ['foo', 'boo']
Bram Moolenaar62aec932022-01-29 21:45:34 +00001228 echo g:FilterWithCond(x, (v) => v =~ '^b')
Bram Moolenaar5f91e742021-03-17 21:29:29 +01001229 endfor
1230
1231 # this fails
1232 lines =<< trim END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001233 echo g:FilterWithCond('foo', (v) => v .. '^b')
Bram Moolenaar5f91e742021-03-17 21:29:29 +01001234 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001235 v9.CheckDefAndScriptFailure(lines, 'E1013: Argument 2: type mismatch, expected func(string): bool but got func(any): string', 1)
Bram Moolenaara9931532021-06-12 15:58:16 +02001236
1237 lines =<< trim END
1238 var Lambda1 = (x) => {
1239 return x
1240 }
1241 assert_equal('asdf', Lambda1('asdf'))
1242 var Lambda2 = (x): string => {
1243 return x
1244 }
1245 assert_equal('foo', Lambda2('foo'))
1246 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001247 v9.CheckDefAndScriptSuccess(lines)
Bram Moolenaara9931532021-06-12 15:58:16 +02001248
1249 lines =<< trim END
1250 var Lambda = (x): string => {
1251 return x
1252 }
1253 echo Lambda(['foo'])
1254 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001255 v9.CheckDefExecAndScriptFailure(lines, 'E1012:')
Bram Moolenaar0346b792021-01-31 22:18:29 +01001256enddef
1257
Bram Moolenaar709664c2020-12-12 14:33:41 +01001258def Test_lambda_uses_assigned_var()
Bram Moolenaar62aec932022-01-29 21:45:34 +00001259 v9.CheckDefSuccess([
Bram Moolenaar709664c2020-12-12 14:33:41 +01001260 'var x: any = "aaa"'
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01001261 'x = filter(["bbb"], (_, v) => v =~ x)'])
Bram Moolenaar709664c2020-12-12 14:33:41 +01001262enddef
1263
Bram Moolenaar18062fc2021-03-05 21:35:47 +01001264def Test_pass_legacy_lambda_to_def_func()
1265 var lines =<< trim END
1266 vim9script
1267 func Foo()
1268 eval s:Bar({x -> 0})
1269 endfunc
1270 def Bar(y: any)
1271 enddef
1272 Foo()
1273 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001274 v9.CheckScriptSuccess(lines)
Bram Moolenaar831bdf82021-06-22 19:32:17 +02001275
1276 lines =<< trim END
1277 vim9script
Bram Moolenaar7a40ff02021-07-04 15:54:08 +02001278 def g:TestFunc(f: func)
Bram Moolenaar831bdf82021-06-22 19:32:17 +02001279 enddef
1280 legacy call g:TestFunc({-> 0})
1281 delfunc g:TestFunc
1282
1283 def g:TestFunc(f: func(number))
1284 enddef
1285 legacy call g:TestFunc({nr -> 0})
1286 delfunc g:TestFunc
1287 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001288 v9.CheckScriptSuccess(lines)
Bram Moolenaar18062fc2021-03-05 21:35:47 +01001289enddef
1290
Bram Moolenaar844fb642021-10-23 13:32:30 +01001291def Test_lambda_in_reduce_line_break()
1292 # this was using freed memory
1293 var lines =<< trim END
1294 vim9script
1295 const result: dict<number> =
1296 ['Bob', 'Sam', 'Cat', 'Bob', 'Cat', 'Cat']
1297 ->reduce((acc, val) => {
1298 if has_key(acc, val)
1299 acc[val] += 1
1300 return acc
1301 else
1302 acc[val] = 1
1303 return acc
1304 endif
1305 }, {})
1306 assert_equal({Bob: 2, Sam: 1, Cat: 3}, result)
1307 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001308 v9.CheckScriptSuccess(lines)
Bram Moolenaar844fb642021-10-23 13:32:30 +01001309enddef
1310
Bram Moolenaardcb53be2021-12-09 14:23:43 +00001311def Test_set_opfunc_to_lambda()
1312 var lines =<< trim END
1313 vim9script
1314 nnoremap <expr> <F4> <SID>CountSpaces() .. '_'
1315 def CountSpaces(type = ''): string
1316 if type == ''
1317 &operatorfunc = (t) => CountSpaces(t)
1318 return 'g@'
1319 endif
1320 normal! '[V']y
1321 g:result = getreg('"')->count(' ')
1322 return ''
1323 enddef
1324 new
1325 'a b c d e'->setline(1)
1326 feedkeys("\<F4>", 'x')
1327 assert_equal(4, g:result)
1328 bwipe!
1329 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001330 v9.CheckScriptSuccess(lines)
Bram Moolenaardcb53be2021-12-09 14:23:43 +00001331enddef
1332
Bram Moolenaaref082e12021-12-12 21:02:03 +00001333def Test_set_opfunc_to_global_function()
1334 var lines =<< trim END
1335 vim9script
1336 def g:CountSpaces(type = ''): string
1337 normal! '[V']y
1338 g:result = getreg('"')->count(' ')
1339 return ''
1340 enddef
Bram Moolenaarb15cf442021-12-16 15:49:43 +00001341 # global function works at script level
Bram Moolenaaref082e12021-12-12 21:02:03 +00001342 &operatorfunc = g:CountSpaces
1343 new
1344 'a b c d e'->setline(1)
1345 feedkeys("g@_", 'x')
1346 assert_equal(4, g:result)
Bram Moolenaarb15cf442021-12-16 15:49:43 +00001347
1348 &operatorfunc = ''
1349 g:result = 0
1350 # global function works in :def function
1351 def Func()
1352 &operatorfunc = g:CountSpaces
1353 enddef
1354 Func()
1355 feedkeys("g@_", 'x')
1356 assert_equal(4, g:result)
1357
Bram Moolenaaref082e12021-12-12 21:02:03 +00001358 bwipe!
1359 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001360 v9.CheckScriptSuccess(lines)
Bram Moolenaaref082e12021-12-12 21:02:03 +00001361 &operatorfunc = ''
1362enddef
1363
Bram Moolenaar33b968d2021-12-13 11:31:04 +00001364def Test_use_script_func_name_with_prefix()
1365 var lines =<< trim END
1366 vim9script
1367 func s:Getit()
1368 return 'it'
1369 endfunc
1370 var Fn = s:Getit
1371 assert_equal('it', Fn())
1372 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001373 v9.CheckScriptSuccess(lines)
Bram Moolenaar33b968d2021-12-13 11:31:04 +00001374enddef
1375
Bram Moolenaardd297bc2021-12-10 10:37:38 +00001376def Test_lambda_type_allocated()
1377 # Check that unreferencing a partial using a lambda can use the variable type
1378 # after the lambda has been freed and does not leak memory.
1379 var lines =<< trim END
1380 vim9script
1381
1382 func MyomniFunc1(val, findstart, base)
1383 return a:findstart ? 0 : []
1384 endfunc
1385
1386 var Lambda = (a, b) => MyomniFunc1(19, a, b)
1387 &omnifunc = Lambda
1388 Lambda = (a, b) => MyomniFunc1(20, a, b)
1389 &omnifunc = string(Lambda)
1390 Lambda = (a, b) => strlen(a)
1391 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001392 v9.CheckScriptSuccess(lines)
Bram Moolenaardd297bc2021-12-10 10:37:38 +00001393enddef
1394
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001395" Default arg and varargs
1396def MyDefVarargs(one: string, two = 'foo', ...rest: list<string>): string
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001397 var res = one .. ',' .. two
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001398 for s in rest
1399 res ..= ',' .. s
1400 endfor
1401 return res
1402enddef
1403
1404def Test_call_def_varargs()
Bram Moolenaar62aec932022-01-29 21:45:34 +00001405 assert_fails('g:MyDefVarargs()', 'E119:', '', 1, 'Test_call_def_varargs')
1406 g:MyDefVarargs('one')->assert_equal('one,foo')
1407 g:MyDefVarargs('one', 'two')->assert_equal('one,two')
1408 g:MyDefVarargs('one', 'two', 'three')->assert_equal('one,two,three')
1409 v9.CheckDefFailure(['g:MyDefVarargs("one", 22)'],
Bram Moolenaar77072282020-09-16 17:55:40 +02001410 'E1013: Argument 2: type mismatch, expected string but got number')
Bram Moolenaar62aec932022-01-29 21:45:34 +00001411 v9.CheckDefFailure(['g:MyDefVarargs("one", "two", 123)'],
Bram Moolenaar77072282020-09-16 17:55:40 +02001412 'E1013: Argument 3: type mismatch, expected string but got number')
Bram Moolenaar24aa48b2020-07-25 16:33:02 +02001413
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001414 var lines =<< trim END
Bram Moolenaar24aa48b2020-07-25 16:33:02 +02001415 vim9script
1416 def Func(...l: list<string>)
1417 echo l
1418 enddef
1419 Func('a', 'b', 'c')
1420 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001421 v9.CheckScriptSuccess(lines)
Bram Moolenaar24aa48b2020-07-25 16:33:02 +02001422
1423 lines =<< trim END
1424 vim9script
1425 def Func(...l: list<string>)
1426 echo l
1427 enddef
1428 Func()
1429 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001430 v9.CheckScriptSuccess(lines)
Bram Moolenaar24aa48b2020-07-25 16:33:02 +02001431
1432 lines =<< trim END
1433 vim9script
Bram Moolenaar2a389082021-04-09 20:24:31 +02001434 def Func(...l: list<any>)
Bram Moolenaar2f8cbc42020-09-16 17:22:59 +02001435 echo l
1436 enddef
1437 Func(0)
1438 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001439 v9.CheckScriptSuccess(lines)
Bram Moolenaar2f8cbc42020-09-16 17:22:59 +02001440
1441 lines =<< trim END
1442 vim9script
Bram Moolenaar2a389082021-04-09 20:24:31 +02001443 def Func(...l: any)
1444 echo l
1445 enddef
1446 Func(0)
1447 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001448 v9.CheckScriptFailure(lines, 'E1180:', 2)
Bram Moolenaar2a389082021-04-09 20:24:31 +02001449
1450 lines =<< trim END
1451 vim9script
Bram Moolenaar28022722020-09-21 22:02:49 +02001452 def Func(..._l: list<string>)
1453 echo _l
1454 enddef
1455 Func('a', 'b', 'c')
1456 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001457 v9.CheckScriptSuccess(lines)
Bram Moolenaar28022722020-09-21 22:02:49 +02001458
1459 lines =<< trim END
1460 vim9script
Bram Moolenaar24aa48b2020-07-25 16:33:02 +02001461 def Func(...l: list<string>)
1462 echo l
1463 enddef
1464 Func(1, 2, 3)
1465 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001466 v9.CheckScriptFailure(lines, 'E1013: Argument 1: type mismatch')
Bram Moolenaar24aa48b2020-07-25 16:33:02 +02001467
1468 lines =<< trim END
1469 vim9script
1470 def Func(...l: list<string>)
1471 echo l
1472 enddef
1473 Func('a', 9)
1474 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001475 v9.CheckScriptFailure(lines, 'E1013: Argument 2: type mismatch')
Bram Moolenaar24aa48b2020-07-25 16:33:02 +02001476
1477 lines =<< trim END
1478 vim9script
1479 def Func(...l: list<string>)
1480 echo l
1481 enddef
1482 Func(1, 'a')
1483 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001484 v9.CheckScriptFailure(lines, 'E1013: Argument 1: type mismatch')
Bram Moolenaar4f53b792021-02-07 15:59:49 +01001485
1486 lines =<< trim END
1487 vim9script
1488 def Func( # some comment
1489 ...l = []
1490 )
1491 echo l
1492 enddef
1493 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001494 v9.CheckScriptFailure(lines, 'E1160:')
Bram Moolenaar6ce46b92021-08-07 15:35:36 +02001495
1496 lines =<< trim END
1497 vim9script
1498 def DoIt()
1499 g:Later('')
1500 enddef
1501 defcompile
1502 def g:Later(...l: list<number>)
1503 enddef
1504 DoIt()
1505 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001506 v9.CheckScriptFailure(lines, 'E1013: Argument 1: type mismatch, expected number but got string')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001507enddef
1508
Bram Moolenaar1378fbc2020-04-11 20:50:33 +02001509let s:value = ''
1510
1511def FuncOneDefArg(opt = 'text')
1512 s:value = opt
1513enddef
1514
1515def FuncTwoDefArg(nr = 123, opt = 'text'): string
1516 return nr .. opt
1517enddef
1518
1519def FuncVarargs(...arg: list<string>): string
1520 return join(arg, ',')
1521enddef
1522
1523def Test_func_type_varargs()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001524 var RefDefArg: func(?string)
Bram Moolenaar848fadd2022-01-30 15:28:30 +00001525 RefDefArg = g:FuncOneDefArg
Bram Moolenaar1378fbc2020-04-11 20:50:33 +02001526 RefDefArg()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001527 s:value->assert_equal('text')
Bram Moolenaar1378fbc2020-04-11 20:50:33 +02001528 RefDefArg('some')
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001529 s:value->assert_equal('some')
Bram Moolenaar1378fbc2020-04-11 20:50:33 +02001530
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001531 var RefDef2Arg: func(?number, ?string): string
Bram Moolenaar848fadd2022-01-30 15:28:30 +00001532 RefDef2Arg = g:FuncTwoDefArg
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001533 RefDef2Arg()->assert_equal('123text')
1534 RefDef2Arg(99)->assert_equal('99text')
1535 RefDef2Arg(77, 'some')->assert_equal('77some')
Bram Moolenaar1378fbc2020-04-11 20:50:33 +02001536
Bram Moolenaar62aec932022-01-29 21:45:34 +00001537 v9.CheckDefFailure(['var RefWrong: func(string?)'], 'E1010:')
1538 v9.CheckDefFailure(['var RefWrong: func(?string, string)'], 'E1007:')
Bram Moolenaar1378fbc2020-04-11 20:50:33 +02001539
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001540 var RefVarargs: func(...list<string>): string
Bram Moolenaar848fadd2022-01-30 15:28:30 +00001541 RefVarargs = g:FuncVarargs
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001542 RefVarargs()->assert_equal('')
1543 RefVarargs('one')->assert_equal('one')
1544 RefVarargs('one', 'two')->assert_equal('one,two')
Bram Moolenaar1378fbc2020-04-11 20:50:33 +02001545
Bram Moolenaar62aec932022-01-29 21:45:34 +00001546 v9.CheckDefFailure(['var RefWrong: func(...list<string>, string)'], 'E110:')
1547 v9.CheckDefFailure(['var RefWrong: func(...list<string>, ?string)'], 'E110:')
Bram Moolenaar1378fbc2020-04-11 20:50:33 +02001548enddef
1549
Bram Moolenaar0b76b422020-04-07 22:05:08 +02001550" Only varargs
1551def MyVarargsOnly(...args: list<string>): string
1552 return join(args, ',')
1553enddef
1554
1555def Test_call_varargs_only()
Bram Moolenaar62aec932022-01-29 21:45:34 +00001556 g:MyVarargsOnly()->assert_equal('')
1557 g:MyVarargsOnly('one')->assert_equal('one')
1558 g:MyVarargsOnly('one', 'two')->assert_equal('one,two')
1559 v9.CheckDefFailure(['g:MyVarargsOnly(1)'], 'E1013: Argument 1: type mismatch, expected string but got number')
1560 v9.CheckDefFailure(['g:MyVarargsOnly("one", 2)'], 'E1013: Argument 2: type mismatch, expected string but got number')
Bram Moolenaar0b76b422020-04-07 22:05:08 +02001561enddef
1562
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001563def Test_using_var_as_arg()
Bram Moolenaard2939812021-12-30 17:09:05 +00001564 var lines =<< trim END
1565 def Func(x: number)
1566 var x = 234
1567 enddef
1568 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001569 v9.CheckDefFailure(lines, 'E1006:')
Bram Moolenaard2939812021-12-30 17:09:05 +00001570
1571 lines =<< trim END
1572 def Func(Ref: number)
1573 def Ref()
1574 enddef
1575 enddef
1576 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001577 v9.CheckDefFailure(lines, 'E1073:')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001578enddef
1579
Bram Moolenaar62aec932022-01-29 21:45:34 +00001580def s:DictArg(arg: dict<string>)
Bram Moolenaarcb2bdb12020-05-10 22:53:56 +02001581 arg['key'] = 'value'
1582enddef
1583
Bram Moolenaar62aec932022-01-29 21:45:34 +00001584def s:ListArg(arg: list<string>)
Bram Moolenaarcb2bdb12020-05-10 22:53:56 +02001585 arg[0] = 'value'
1586enddef
1587
1588def Test_assign_to_argument()
Bram Moolenaarf5be8cd2020-07-17 20:36:00 +02001589 # works for dict and list
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001590 var d: dict<string> = {}
Bram Moolenaarcb2bdb12020-05-10 22:53:56 +02001591 DictArg(d)
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001592 d['key']->assert_equal('value')
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001593 var l: list<string> = []
Bram Moolenaarcb2bdb12020-05-10 22:53:56 +02001594 ListArg(l)
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001595 l[0]->assert_equal('value')
Bram Moolenaarcb2bdb12020-05-10 22:53:56 +02001596
Bram Moolenaar62aec932022-01-29 21:45:34 +00001597 v9.CheckScriptFailure(['def Func(arg: number)', 'arg = 3', 'enddef', 'defcompile'], 'E1090:')
Bram Moolenaar2d870f82020-12-05 13:41:01 +01001598 delfunc! g:Func
Bram Moolenaarcb2bdb12020-05-10 22:53:56 +02001599enddef
1600
Bram Moolenaarb816dae2020-09-20 22:04:00 +02001601" These argument names are reserved in legacy functions.
Bram Moolenaar62aec932022-01-29 21:45:34 +00001602def s:WithReservedNames(firstline: string, lastline: string): string
Bram Moolenaarb816dae2020-09-20 22:04:00 +02001603 return firstline .. lastline
1604enddef
1605
1606def Test_argument_names()
1607 assert_equal('OK', WithReservedNames('O', 'K'))
1608enddef
1609
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001610def Test_call_func_defined_later()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001611 g:DefinedLater('one')->assert_equal('one')
Bram Moolenaar9bd5d872020-09-06 21:47:48 +02001612 assert_fails('NotDefined("one")', 'E117:', '', 2, 'Test_call_func_defined_later')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001613enddef
1614
Bram Moolenaar1df8b3f2020-04-23 18:13:23 +02001615func DefinedLater(arg)
1616 return a:arg
1617endfunc
1618
1619def Test_call_funcref()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001620 g:SomeFunc('abc')->assert_equal(3)
Bram Moolenaar9bd5d872020-09-06 21:47:48 +02001621 assert_fails('NotAFunc()', 'E117:', '', 2, 'Test_call_funcref') # comment after call
Bram Moolenaar2ef91562021-12-11 16:14:07 +00001622 assert_fails('g:NotAFunc()', 'E1085:', '', 3, 'Test_call_funcref')
Bram Moolenaar2f1980f2020-07-22 19:30:06 +02001623
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001624 var lines =<< trim END
Bram Moolenaar2f1980f2020-07-22 19:30:06 +02001625 vim9script
1626 def RetNumber(): number
1627 return 123
1628 enddef
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001629 var Funcref: func: number = function('RetNumber')
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001630 Funcref()->assert_equal(123)
Bram Moolenaar2f1980f2020-07-22 19:30:06 +02001631 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001632 v9.CheckScriptSuccess(lines)
Bram Moolenaar0f60e802020-07-22 20:16:11 +02001633
1634 lines =<< trim END
1635 vim9script
1636 def RetNumber(): number
1637 return 123
1638 enddef
1639 def Bar(F: func: number): number
1640 return F()
1641 enddef
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001642 var Funcref = function('RetNumber')
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001643 Bar(Funcref)->assert_equal(123)
Bram Moolenaar0f60e802020-07-22 20:16:11 +02001644 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001645 v9.CheckScriptSuccess(lines)
Bram Moolenaarbfba8652020-07-23 20:09:10 +02001646
1647 lines =<< trim END
1648 vim9script
1649 def UseNumber(nr: number)
1650 echo nr
1651 enddef
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001652 var Funcref: func(number) = function('UseNumber')
Bram Moolenaarbfba8652020-07-23 20:09:10 +02001653 Funcref(123)
1654 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001655 v9.CheckScriptSuccess(lines)
Bram Moolenaarb8070e32020-07-23 20:56:04 +02001656
1657 lines =<< trim END
1658 vim9script
1659 def UseNumber(nr: number)
1660 echo nr
1661 enddef
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001662 var Funcref: func(string) = function('UseNumber')
Bram Moolenaarb8070e32020-07-23 20:56:04 +02001663 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001664 v9.CheckScriptFailure(lines, 'E1012: Type mismatch; expected func(string) but got func(number)')
Bram Moolenaar4fc224c2020-07-26 17:56:25 +02001665
1666 lines =<< trim END
1667 vim9script
1668 def EchoNr(nr = 34)
1669 g:echo = nr
1670 enddef
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001671 var Funcref: func(?number) = function('EchoNr')
Bram Moolenaar4fc224c2020-07-26 17:56:25 +02001672 Funcref()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001673 g:echo->assert_equal(34)
Bram Moolenaar4fc224c2020-07-26 17:56:25 +02001674 Funcref(123)
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001675 g:echo->assert_equal(123)
Bram Moolenaar4fc224c2020-07-26 17:56:25 +02001676 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001677 v9.CheckScriptSuccess(lines)
Bram Moolenaarace61322020-07-26 18:16:58 +02001678
1679 lines =<< trim END
1680 vim9script
1681 def EchoList(...l: list<number>)
1682 g:echo = l
1683 enddef
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001684 var Funcref: func(...list<number>) = function('EchoList')
Bram Moolenaarace61322020-07-26 18:16:58 +02001685 Funcref()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001686 g:echo->assert_equal([])
Bram Moolenaarace61322020-07-26 18:16:58 +02001687 Funcref(1, 2, 3)
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001688 g:echo->assert_equal([1, 2, 3])
Bram Moolenaarace61322020-07-26 18:16:58 +02001689 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001690 v9.CheckScriptSuccess(lines)
Bram Moolenaar01865ad2020-07-26 18:33:09 +02001691
1692 lines =<< trim END
1693 vim9script
1694 def OptAndVar(nr: number, opt = 12, ...l: list<number>): number
1695 g:optarg = opt
1696 g:listarg = l
1697 return nr
1698 enddef
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001699 var Funcref: func(number, ?number, ...list<number>): number = function('OptAndVar')
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001700 Funcref(10)->assert_equal(10)
1701 g:optarg->assert_equal(12)
1702 g:listarg->assert_equal([])
Bram Moolenaar01865ad2020-07-26 18:33:09 +02001703
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001704 Funcref(11, 22)->assert_equal(11)
1705 g:optarg->assert_equal(22)
1706 g:listarg->assert_equal([])
Bram Moolenaar01865ad2020-07-26 18:33:09 +02001707
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001708 Funcref(17, 18, 1, 2, 3)->assert_equal(17)
1709 g:optarg->assert_equal(18)
1710 g:listarg->assert_equal([1, 2, 3])
Bram Moolenaar01865ad2020-07-26 18:33:09 +02001711 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001712 v9.CheckScriptSuccess(lines)
Bram Moolenaar1df8b3f2020-04-23 18:13:23 +02001713enddef
1714
1715let SomeFunc = function('len')
1716let NotAFunc = 'text'
1717
Bram Moolenaar99aaf0c2020-04-12 14:39:53 +02001718def CombineFuncrefTypes()
Bram Moolenaarf5be8cd2020-07-17 20:36:00 +02001719 # same arguments, different return type
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001720 var Ref1: func(bool): string
1721 var Ref2: func(bool): number
1722 var Ref3: func(bool): any
Bram Moolenaar99aaf0c2020-04-12 14:39:53 +02001723 Ref3 = g:cond ? Ref1 : Ref2
1724
Bram Moolenaarf5be8cd2020-07-17 20:36:00 +02001725 # different number of arguments
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001726 var Refa1: func(bool): number
1727 var Refa2: func(bool, number): number
1728 var Refa3: func: number
Bram Moolenaar99aaf0c2020-04-12 14:39:53 +02001729 Refa3 = g:cond ? Refa1 : Refa2
1730
Bram Moolenaarf5be8cd2020-07-17 20:36:00 +02001731 # different argument types
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001732 var Refb1: func(bool, string): number
1733 var Refb2: func(string, number): number
1734 var Refb3: func(any, any): number
Bram Moolenaar99aaf0c2020-04-12 14:39:53 +02001735 Refb3 = g:cond ? Refb1 : Refb2
1736enddef
1737
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001738def FuncWithForwardCall()
Bram Moolenaar1df8b3f2020-04-23 18:13:23 +02001739 return g:DefinedEvenLater("yes")
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001740enddef
1741
1742def DefinedEvenLater(arg: string): string
1743 return arg
1744enddef
1745
1746def Test_error_in_nested_function()
Bram Moolenaarf5be8cd2020-07-17 20:36:00 +02001747 # Error in called function requires unwinding the call stack.
Bram Moolenaar848fadd2022-01-30 15:28:30 +00001748 assert_fails('g:FuncWithForwardCall()', 'E1096:', '', 1, 'FuncWithForwardCall')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001749enddef
1750
Bram Moolenaar4bf10062021-12-28 17:23:12 +00001751def Test_nested_function_with_nextcmd()
Bram Moolenaar9c23f9b2021-12-26 14:23:22 +00001752 var lines =<< trim END
1753 vim9script
1754 # Define an outer function
1755 def FirstFunction()
1756 # Define an inner function
1757 def SecondFunction()
1758 # the function has a body, a double free is detected.
1759 AAAAA
1760
1761 # enddef followed by | or } followed by # one or more characters
1762 enddef|BBBB
1763 enddef
1764
1765 # Compile all functions
1766 defcompile
1767 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001768 v9.CheckScriptFailure(lines, 'E1173: Text found after enddef: BBBB')
Bram Moolenaar9c23f9b2021-12-26 14:23:22 +00001769enddef
1770
Bram Moolenaar4bf10062021-12-28 17:23:12 +00001771def Test_nested_function_with_args_split()
1772 var lines =<< trim END
1773 vim9script
1774 def FirstFunction()
1775 def SecondFunction(
1776 )
1777 # had a double free if the right parenthesis of the nested function is
1778 # on the next line
1779
1780 enddef|BBBB
1781 enddef
1782 # Compile all functions
1783 defcompile
1784 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001785 v9.CheckScriptFailure(lines, 'E1173: Text found after enddef: BBBB')
Bram Moolenaar7473a842021-12-28 17:55:26 +00001786
1787 lines =<< trim END
1788 vim9script
1789 def FirstFunction()
1790 func SecondFunction()
1791 endfunc|BBBB
1792 enddef
1793 defcompile
1794 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001795 v9.CheckScriptFailure(lines, 'E1173: Text found after endfunction: BBBB')
Bram Moolenaar4bf10062021-12-28 17:23:12 +00001796enddef
1797
Bram Moolenaar9f1a39a2022-01-08 15:39:39 +00001798def Test_error_in_function_args()
1799 var lines =<< trim END
1800 def FirstFunction()
1801 def SecondFunction(J =
1802 # Nois
1803 # one
1804
1805 enddef|BBBB
1806 enddef
1807 # Compile all functions
1808 defcompile
1809 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001810 v9.CheckScriptFailure(lines, 'E488:')
Bram Moolenaar9f1a39a2022-01-08 15:39:39 +00001811enddef
1812
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001813def Test_return_type_wrong()
Bram Moolenaar62aec932022-01-29 21:45:34 +00001814 v9.CheckScriptFailure([
Bram Moolenaar5a849da2020-08-08 16:47:30 +02001815 'def Func(): number',
1816 'return "a"',
1817 'enddef',
1818 'defcompile'], 'expected number but got string')
Bram Moolenaar2d870f82020-12-05 13:41:01 +01001819 delfunc! g:Func
Bram Moolenaar62aec932022-01-29 21:45:34 +00001820 v9.CheckScriptFailure([
Bram Moolenaar5a849da2020-08-08 16:47:30 +02001821 'def Func(): string',
1822 'return 1',
1823 'enddef',
1824 'defcompile'], 'expected string but got number')
Bram Moolenaar2d870f82020-12-05 13:41:01 +01001825 delfunc! g:Func
Bram Moolenaar62aec932022-01-29 21:45:34 +00001826 v9.CheckScriptFailure([
Bram Moolenaar5a849da2020-08-08 16:47:30 +02001827 'def Func(): void',
1828 'return "a"',
1829 'enddef',
1830 'defcompile'],
1831 'E1096: Returning a value in a function without a return type')
Bram Moolenaar2d870f82020-12-05 13:41:01 +01001832 delfunc! g:Func
Bram Moolenaar62aec932022-01-29 21:45:34 +00001833 v9.CheckScriptFailure([
Bram Moolenaar5a849da2020-08-08 16:47:30 +02001834 'def Func()',
1835 'return "a"',
1836 'enddef',
1837 'defcompile'],
1838 'E1096: Returning a value in a function without a return type')
Bram Moolenaar2d870f82020-12-05 13:41:01 +01001839 delfunc! g:Func
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001840
Bram Moolenaar62aec932022-01-29 21:45:34 +00001841 v9.CheckScriptFailure([
Bram Moolenaar5a849da2020-08-08 16:47:30 +02001842 'def Func(): number',
1843 'return',
1844 'enddef',
1845 'defcompile'], 'E1003:')
Bram Moolenaar2d870f82020-12-05 13:41:01 +01001846 delfunc! g:Func
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001847
Bram Moolenaar62aec932022-01-29 21:45:34 +00001848 v9.CheckScriptFailure([
Bram Moolenaar33ea9fd2021-08-08 19:07:37 +02001849 'def Func():number',
1850 'return 123',
1851 'enddef',
1852 'defcompile'], 'E1069:')
1853 delfunc! g:Func
1854
Bram Moolenaar62aec932022-01-29 21:45:34 +00001855 v9.CheckScriptFailure([
Bram Moolenaar33ea9fd2021-08-08 19:07:37 +02001856 'def Func() :number',
1857 'return 123',
1858 'enddef',
1859 'defcompile'], 'E1059:')
1860 delfunc! g:Func
1861
Bram Moolenaar62aec932022-01-29 21:45:34 +00001862 v9.CheckScriptFailure([
Bram Moolenaar33ea9fd2021-08-08 19:07:37 +02001863 'def Func() : number',
1864 'return 123',
1865 'enddef',
1866 'defcompile'], 'E1059:')
1867 delfunc! g:Func
1868
Bram Moolenaar62aec932022-01-29 21:45:34 +00001869 v9.CheckScriptFailure(['def Func(): list', 'return []', 'enddef'], 'E1008:')
Bram Moolenaar2d870f82020-12-05 13:41:01 +01001870 delfunc! g:Func
Bram Moolenaar62aec932022-01-29 21:45:34 +00001871 v9.CheckScriptFailure(['def Func(): dict', 'return {}', 'enddef'], 'E1008:')
Bram Moolenaar2d870f82020-12-05 13:41:01 +01001872 delfunc! g:Func
Bram Moolenaar62aec932022-01-29 21:45:34 +00001873 v9.CheckScriptFailure(['def Func()', 'return 1'], 'E1057:')
Bram Moolenaar2d870f82020-12-05 13:41:01 +01001874 delfunc! g:Func
Bram Moolenaar5a849da2020-08-08 16:47:30 +02001875
Bram Moolenaar62aec932022-01-29 21:45:34 +00001876 v9.CheckScriptFailure([
Bram Moolenaar5a849da2020-08-08 16:47:30 +02001877 'vim9script',
1878 'def FuncB()',
1879 ' return 123',
1880 'enddef',
1881 'def FuncA()',
1882 ' FuncB()',
1883 'enddef',
1884 'defcompile'], 'E1096:')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001885enddef
1886
1887def Test_arg_type_wrong()
Bram Moolenaar62aec932022-01-29 21:45:34 +00001888 v9.CheckScriptFailure(['def Func3(items: list)', 'echo "a"', 'enddef'], 'E1008: Missing <type>')
1889 v9.CheckScriptFailure(['def Func4(...)', 'echo "a"', 'enddef'], 'E1055: Missing name after ...')
1890 v9.CheckScriptFailure(['def Func5(items:string)', 'echo "a"'], 'E1069:')
1891 v9.CheckScriptFailure(['def Func5(items)', 'echo "a"'], 'E1077:')
1892 v9.CheckScriptFailure(['def Func6(...x:list<number>)', 'echo "a"', 'enddef'], 'E1069:')
1893 v9.CheckScriptFailure(['def Func7(...x: int)', 'echo "a"', 'enddef'], 'E1010:')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001894enddef
1895
Bram Moolenaar86cdb8a2021-04-06 19:01:03 +02001896def Test_white_space_before_comma()
1897 var lines =<< trim END
1898 vim9script
1899 def Func(a: number , b: number)
1900 enddef
1901 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001902 v9.CheckScriptFailure(lines, 'E1068:')
Yegappan Lakshmanan611728f2021-05-24 15:15:47 +02001903 call assert_fails('vim9cmd echo stridx("a" .. "b" , "a")', 'E1068:')
Bram Moolenaar86cdb8a2021-04-06 19:01:03 +02001904enddef
1905
Bram Moolenaar608d78f2021-03-06 22:33:12 +01001906def Test_white_space_after_comma()
1907 var lines =<< trim END
1908 vim9script
1909 def Func(a: number,b: number)
1910 enddef
1911 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001912 v9.CheckScriptFailure(lines, 'E1069:')
Bram Moolenaar608d78f2021-03-06 22:33:12 +01001913
1914 # OK in legacy function
1915 lines =<< trim END
1916 vim9script
1917 func Func(a,b)
1918 endfunc
1919 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001920 v9.CheckScriptSuccess(lines)
Bram Moolenaar608d78f2021-03-06 22:33:12 +01001921enddef
1922
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001923def Test_vim9script_call()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001924 var lines =<< trim END
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001925 vim9script
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001926 var name = ''
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001927 def MyFunc(arg: string)
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001928 name = arg
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001929 enddef
1930 MyFunc('foobar')
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001931 name->assert_equal('foobar')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001932
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001933 var str = 'barfoo'
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001934 str->MyFunc()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001935 name->assert_equal('barfoo')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001936
Bram Moolenaar67979662020-06-20 22:50:47 +02001937 g:value = 'value'
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001938 g:value->MyFunc()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001939 name->assert_equal('value')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001940
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001941 var listvar = []
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001942 def ListFunc(arg: list<number>)
1943 listvar = arg
1944 enddef
1945 [1, 2, 3]->ListFunc()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001946 listvar->assert_equal([1, 2, 3])
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001947
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001948 var dictvar = {}
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001949 def DictFunc(arg: dict<number>)
1950 dictvar = arg
1951 enddef
Bram Moolenaare0de1712020-12-02 17:36:54 +01001952 {a: 1, b: 2}->DictFunc()
1953 dictvar->assert_equal({a: 1, b: 2})
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001954 def CompiledDict()
Bram Moolenaare0de1712020-12-02 17:36:54 +01001955 {a: 3, b: 4}->DictFunc()
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001956 enddef
1957 CompiledDict()
Bram Moolenaare0de1712020-12-02 17:36:54 +01001958 dictvar->assert_equal({a: 3, b: 4})
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001959
Bram Moolenaare0de1712020-12-02 17:36:54 +01001960 {a: 3, b: 4}->DictFunc()
1961 dictvar->assert_equal({a: 3, b: 4})
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001962
1963 ('text')->MyFunc()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001964 name->assert_equal('text')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001965 ("some")->MyFunc()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001966 name->assert_equal('some')
Bram Moolenaare6b53242020-07-01 17:28:33 +02001967
Bram Moolenaar13e12b82020-07-24 18:47:22 +02001968 # line starting with single quote is not a mark
Bram Moolenaar10409562020-07-29 20:00:38 +02001969 # line starting with double quote can be a method call
Bram Moolenaar3d48e252020-07-15 14:15:52 +02001970 'asdfasdf'->MyFunc()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001971 name->assert_equal('asdfasdf')
Bram Moolenaar10409562020-07-29 20:00:38 +02001972 "xyz"->MyFunc()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001973 name->assert_equal('xyz')
Bram Moolenaar3d48e252020-07-15 14:15:52 +02001974
1975 def UseString()
1976 'xyork'->MyFunc()
1977 enddef
1978 UseString()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001979 name->assert_equal('xyork')
Bram Moolenaar3d48e252020-07-15 14:15:52 +02001980
Bram Moolenaar10409562020-07-29 20:00:38 +02001981 def UseString2()
1982 "knife"->MyFunc()
1983 enddef
1984 UseString2()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001985 name->assert_equal('knife')
Bram Moolenaar10409562020-07-29 20:00:38 +02001986
Bram Moolenaar13e12b82020-07-24 18:47:22 +02001987 # prepending a colon makes it a mark
1988 new
1989 setline(1, ['aaa', 'bbb', 'ccc'])
1990 normal! 3Gmt1G
1991 :'t
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001992 getcurpos()[1]->assert_equal(3)
Bram Moolenaar13e12b82020-07-24 18:47:22 +02001993 bwipe!
1994
Bram Moolenaare6b53242020-07-01 17:28:33 +02001995 MyFunc(
1996 'continued'
1997 )
1998 assert_equal('continued',
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001999 name
Bram Moolenaare6b53242020-07-01 17:28:33 +02002000 )
2001
2002 call MyFunc(
2003 'more'
2004 ..
2005 'lines'
2006 )
2007 assert_equal(
2008 'morelines',
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002009 name)
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002010 END
2011 writefile(lines, 'Xcall.vim')
2012 source Xcall.vim
2013 delete('Xcall.vim')
2014enddef
2015
2016def Test_vim9script_call_fail_decl()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002017 var lines =<< trim END
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002018 vim9script
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002019 var name = ''
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002020 def MyFunc(arg: string)
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002021 var name = 123
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002022 enddef
Bram Moolenaar822ba242020-05-24 23:00:18 +02002023 defcompile
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002024 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002025 v9.CheckScriptFailure(lines, 'E1054:')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002026enddef
2027
Bram Moolenaar65b95452020-07-19 14:03:09 +02002028def Test_vim9script_call_fail_type()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002029 var lines =<< trim END
Bram Moolenaar65b95452020-07-19 14:03:09 +02002030 vim9script
2031 def MyFunc(arg: string)
2032 echo arg
2033 enddef
2034 MyFunc(1234)
2035 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002036 v9.CheckScriptFailure(lines, 'E1013: Argument 1: type mismatch, expected string but got number')
Bram Moolenaar65b95452020-07-19 14:03:09 +02002037enddef
2038
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002039def Test_vim9script_call_fail_const()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002040 var lines =<< trim END
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002041 vim9script
2042 const var = ''
2043 def MyFunc(arg: string)
2044 var = 'asdf'
2045 enddef
Bram Moolenaar822ba242020-05-24 23:00:18 +02002046 defcompile
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002047 END
2048 writefile(lines, 'Xcall_const.vim')
Bram Moolenaar9bd5d872020-09-06 21:47:48 +02002049 assert_fails('source Xcall_const.vim', 'E46:', '', 1, 'MyFunc')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002050 delete('Xcall_const.vim')
Bram Moolenaar3bdc90b2020-12-22 20:35:40 +01002051
2052 lines =<< trim END
2053 const g:Aconst = 77
2054 def Change()
2055 # comment
2056 g:Aconst = 99
2057 enddef
2058 call Change()
2059 unlet g:Aconst
2060 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002061 v9.CheckScriptFailure(lines, 'E741: Value is locked: Aconst', 2)
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002062enddef
2063
2064" Test that inside :function a Python function can be defined, :def is not
2065" recognized.
2066func Test_function_python()
2067 CheckFeature python3
Bram Moolenaar727345e2020-09-27 23:33:59 +02002068 let py = 'python3'
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002069 execute py "<< EOF"
2070def do_something():
2071 return 1
2072EOF
2073endfunc
2074
2075def Test_delfunc()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002076 var lines =<< trim END
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002077 vim9script
Bram Moolenaar4c17ad92020-04-27 22:47:51 +02002078 def g:GoneSoon()
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002079 echo 'hello'
2080 enddef
2081
2082 def CallGoneSoon()
Bram Moolenaar62aec932022-01-29 21:45:34 +00002083 g:GoneSoon()
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002084 enddef
Bram Moolenaar822ba242020-05-24 23:00:18 +02002085 defcompile
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002086
Bram Moolenaar4c17ad92020-04-27 22:47:51 +02002087 delfunc g:GoneSoon
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002088 CallGoneSoon()
2089 END
2090 writefile(lines, 'XToDelFunc')
Bram Moolenaar9bd5d872020-09-06 21:47:48 +02002091 assert_fails('so XToDelFunc', 'E933:', '', 1, 'CallGoneSoon')
2092 assert_fails('so XToDelFunc', 'E933:', '', 1, 'CallGoneSoon')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002093
2094 delete('XToDelFunc')
2095enddef
2096
Bram Moolenaar7509ad82021-12-14 18:14:37 +00002097func Test_free_dict_while_in_funcstack()
2098 " relies on the sleep command
2099 CheckUnix
2100 call Run_Test_free_dict_while_in_funcstack()
2101endfunc
2102
2103def Run_Test_free_dict_while_in_funcstack()
Bram Moolenaar7509ad82021-12-14 18:14:37 +00002104 # this was freeing the TermRun() default argument dictionary while it was
2105 # still referenced in a funcstack_T
2106 var lines =<< trim END
2107 vim9script
2108
2109 &updatetime = 400
2110 def TermRun(_ = {})
2111 def Post()
2112 enddef
2113 def Exec()
2114 term_start('sleep 1', {
2115 term_finish: 'close',
2116 exit_cb: (_, _) => Post(),
2117 })
2118 enddef
2119 Exec()
2120 enddef
2121 nnoremap <F4> <Cmd>call <SID>TermRun()<CR>
2122 timer_start(100, (_) => feedkeys("\<F4>"))
2123 timer_start(1000, (_) => feedkeys("\<F4>"))
2124 sleep 1500m
2125 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002126 v9.CheckScriptSuccess(lines)
Bram Moolenaar7509ad82021-12-14 18:14:37 +00002127 nunmap <F4>
2128 set updatetime&
2129enddef
2130
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002131def Test_redef_failure()
Bram Moolenaard2c61702020-09-06 15:58:36 +02002132 writefile(['def Func0(): string', 'return "Func0"', 'enddef'], 'Xdef')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002133 so Xdef
Bram Moolenaard2c61702020-09-06 15:58:36 +02002134 writefile(['def Func1(): string', 'return "Func1"', 'enddef'], 'Xdef')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002135 so Xdef
Bram Moolenaard2c61702020-09-06 15:58:36 +02002136 writefile(['def! Func0(): string', 'enddef', 'defcompile'], 'Xdef')
Bram Moolenaar9bd5d872020-09-06 21:47:48 +02002137 assert_fails('so Xdef', 'E1027:', '', 1, 'Func0')
Bram Moolenaard2c61702020-09-06 15:58:36 +02002138 writefile(['def Func2(): string', 'return "Func2"', 'enddef'], 'Xdef')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002139 so Xdef
Bram Moolenaard2c61702020-09-06 15:58:36 +02002140 delete('Xdef')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002141
Bram Moolenaar701cc6c2021-04-10 13:33:48 +02002142 assert_fails('g:Func0()', 'E1091:')
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002143 g:Func1()->assert_equal('Func1')
2144 g:Func2()->assert_equal('Func2')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002145
2146 delfunc! Func0
2147 delfunc! Func1
2148 delfunc! Func2
2149enddef
2150
Bram Moolenaarf93c7fe2020-04-23 22:16:53 +02002151def Test_vim9script_func()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002152 var lines =<< trim END
Bram Moolenaarf93c7fe2020-04-23 22:16:53 +02002153 vim9script
2154 func Func(arg)
2155 echo a:arg
2156 endfunc
2157 Func('text')
2158 END
2159 writefile(lines, 'XVim9Func')
2160 so XVim9Func
2161
2162 delete('XVim9Func')
2163enddef
2164
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002165let s:funcResult = 0
2166
2167def FuncNoArgNoRet()
Bram Moolenaar53900992020-08-22 19:02:02 +02002168 s:funcResult = 11
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002169enddef
2170
2171def FuncNoArgRetNumber(): number
Bram Moolenaar53900992020-08-22 19:02:02 +02002172 s:funcResult = 22
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002173 return 1234
2174enddef
2175
Bram Moolenaarec5929d2020-04-07 20:53:39 +02002176def FuncNoArgRetString(): string
Bram Moolenaar53900992020-08-22 19:02:02 +02002177 s:funcResult = 45
Bram Moolenaarec5929d2020-04-07 20:53:39 +02002178 return 'text'
2179enddef
2180
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002181def FuncOneArgNoRet(arg: number)
Bram Moolenaar53900992020-08-22 19:02:02 +02002182 s:funcResult = arg
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002183enddef
2184
2185def FuncOneArgRetNumber(arg: number): number
Bram Moolenaar53900992020-08-22 19:02:02 +02002186 s:funcResult = arg
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002187 return arg
2188enddef
2189
Bram Moolenaar08938ee2020-04-11 23:17:17 +02002190def FuncTwoArgNoRet(one: bool, two: number)
Bram Moolenaar53900992020-08-22 19:02:02 +02002191 s:funcResult = two
Bram Moolenaar08938ee2020-04-11 23:17:17 +02002192enddef
2193
Bram Moolenaar62aec932022-01-29 21:45:34 +00002194def s:FuncOneArgRetString(arg: string): string
Bram Moolenaarec5929d2020-04-07 20:53:39 +02002195 return arg
2196enddef
2197
Bram Moolenaar62aec932022-01-29 21:45:34 +00002198def s:FuncOneArgRetAny(arg: any): any
Bram Moolenaar89228602020-04-05 22:14:54 +02002199 return arg
2200enddef
2201
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002202def Test_func_type()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002203 var Ref1: func()
Bram Moolenaar53900992020-08-22 19:02:02 +02002204 s:funcResult = 0
Bram Moolenaar62aec932022-01-29 21:45:34 +00002205 Ref1 = g:FuncNoArgNoRet
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002206 Ref1()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002207 s:funcResult->assert_equal(11)
Bram Moolenaar4c683752020-04-05 21:38:23 +02002208
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002209 var Ref2: func
Bram Moolenaar53900992020-08-22 19:02:02 +02002210 s:funcResult = 0
Bram Moolenaar62aec932022-01-29 21:45:34 +00002211 Ref2 = g:FuncNoArgNoRet
Bram Moolenaar4c683752020-04-05 21:38:23 +02002212 Ref2()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002213 s:funcResult->assert_equal(11)
Bram Moolenaar4c683752020-04-05 21:38:23 +02002214
Bram Moolenaar53900992020-08-22 19:02:02 +02002215 s:funcResult = 0
Bram Moolenaar62aec932022-01-29 21:45:34 +00002216 Ref2 = g:FuncOneArgNoRet
Bram Moolenaar4c683752020-04-05 21:38:23 +02002217 Ref2(12)
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002218 s:funcResult->assert_equal(12)
Bram Moolenaar4c683752020-04-05 21:38:23 +02002219
Bram Moolenaar53900992020-08-22 19:02:02 +02002220 s:funcResult = 0
Bram Moolenaar62aec932022-01-29 21:45:34 +00002221 Ref2 = g:FuncNoArgRetNumber
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002222 Ref2()->assert_equal(1234)
2223 s:funcResult->assert_equal(22)
Bram Moolenaar4c683752020-04-05 21:38:23 +02002224
Bram Moolenaar53900992020-08-22 19:02:02 +02002225 s:funcResult = 0
Bram Moolenaar848fadd2022-01-30 15:28:30 +00002226 Ref2 = g:FuncOneArgRetNumber
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002227 Ref2(13)->assert_equal(13)
2228 s:funcResult->assert_equal(13)
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002229enddef
2230
Bram Moolenaar9978d472020-07-05 16:01:56 +02002231def Test_repeat_return_type()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002232 var res = 0
Bram Moolenaar9978d472020-07-05 16:01:56 +02002233 for n in repeat([1], 3)
2234 res += n
2235 endfor
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002236 res->assert_equal(3)
Bram Moolenaarfce82b32020-07-05 16:07:21 +02002237
2238 res = 0
2239 for n in add([1, 2], 3)
2240 res += n
2241 endfor
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002242 res->assert_equal(6)
Bram Moolenaar9978d472020-07-05 16:01:56 +02002243enddef
2244
Bram Moolenaar846178a2020-07-05 17:04:13 +02002245def Test_argv_return_type()
2246 next fileone filetwo
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002247 var res = ''
Bram Moolenaar846178a2020-07-05 17:04:13 +02002248 for name in argv()
2249 res ..= name
2250 endfor
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002251 res->assert_equal('fileonefiletwo')
Bram Moolenaar846178a2020-07-05 17:04:13 +02002252enddef
2253
Bram Moolenaarec5929d2020-04-07 20:53:39 +02002254def Test_func_type_part()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002255 var RefVoid: func: void
Bram Moolenaar62aec932022-01-29 21:45:34 +00002256 RefVoid = g:FuncNoArgNoRet
2257 RefVoid = g:FuncOneArgNoRet
2258 v9.CheckDefFailure(['var RefVoid: func: void', 'RefVoid = g:FuncNoArgRetNumber'], 'E1012: Type mismatch; expected func(...) but got func(): number')
Bram Moolenaar848fadd2022-01-30 15:28:30 +00002259 v9.CheckDefFailure(['var RefVoid: func: void', 'RefVoid = g:FuncNoArgRetString'], 'E1012: Type mismatch; expected func(...) but got func(): string')
Bram Moolenaarec5929d2020-04-07 20:53:39 +02002260
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002261 var RefAny: func(): any
Bram Moolenaar62aec932022-01-29 21:45:34 +00002262 RefAny = g:FuncNoArgRetNumber
Bram Moolenaar848fadd2022-01-30 15:28:30 +00002263 RefAny = g:FuncNoArgRetString
Bram Moolenaar62aec932022-01-29 21:45:34 +00002264 v9.CheckDefFailure(['var RefAny: func(): any', 'RefAny = g:FuncNoArgNoRet'], 'E1012: Type mismatch; expected func(): any but got func()')
2265 v9.CheckDefFailure(['var RefAny: func(): any', 'RefAny = g:FuncOneArgNoRet'], 'E1012: Type mismatch; expected func(): any but got func(number)')
Bram Moolenaarec5929d2020-04-07 20:53:39 +02002266
Bram Moolenaar6abd3dc2020-10-04 14:17:32 +02002267 var RefAnyNoArgs: func: any = RefAny
2268
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002269 var RefNr: func: number
Bram Moolenaar62aec932022-01-29 21:45:34 +00002270 RefNr = g:FuncNoArgRetNumber
Bram Moolenaar848fadd2022-01-30 15:28:30 +00002271 RefNr = g:FuncOneArgRetNumber
Bram Moolenaar62aec932022-01-29 21:45:34 +00002272 v9.CheckDefFailure(['var RefNr: func: number', 'RefNr = g:FuncNoArgNoRet'], 'E1012: Type mismatch; expected func(...): number but got func()')
Bram Moolenaar848fadd2022-01-30 15:28:30 +00002273 v9.CheckDefFailure(['var RefNr: func: number', 'RefNr = g:FuncNoArgRetString'], 'E1012: Type mismatch; expected func(...): number but got func(): string')
Bram Moolenaarec5929d2020-04-07 20:53:39 +02002274
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002275 var RefStr: func: string
Bram Moolenaar848fadd2022-01-30 15:28:30 +00002276 RefStr = g:FuncNoArgRetString
Bram Moolenaarec5929d2020-04-07 20:53:39 +02002277 RefStr = FuncOneArgRetString
Bram Moolenaar62aec932022-01-29 21:45:34 +00002278 v9.CheckDefFailure(['var RefStr: func: string', 'RefStr = g:FuncNoArgNoRet'], 'E1012: Type mismatch; expected func(...): string but got func()')
2279 v9.CheckDefFailure(['var RefStr: func: string', 'RefStr = g:FuncNoArgRetNumber'], 'E1012: Type mismatch; expected func(...): string but got func(): number')
Bram Moolenaarec5929d2020-04-07 20:53:39 +02002280enddef
2281
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002282def Test_func_type_fails()
Bram Moolenaar62aec932022-01-29 21:45:34 +00002283 v9.CheckDefFailure(['var ref1: func()'], 'E704:')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002284
Bram Moolenaar62aec932022-01-29 21:45:34 +00002285 v9.CheckDefFailure(['var Ref1: func()', 'Ref1 = g:FuncNoArgRetNumber'], 'E1012: Type mismatch; expected func() but got func(): number')
2286 v9.CheckDefFailure(['var Ref1: func()', 'Ref1 = g:FuncOneArgNoRet'], 'E1012: Type mismatch; expected func() but got func(number)')
Bram Moolenaar848fadd2022-01-30 15:28:30 +00002287 v9.CheckDefFailure(['var Ref1: func()', 'Ref1 = g:FuncOneArgRetNumber'], 'E1012: Type mismatch; expected func() but got func(number): number')
Bram Moolenaar62aec932022-01-29 21:45:34 +00002288 v9.CheckDefFailure(['var Ref1: func(bool)', 'Ref1 = g:FuncTwoArgNoRet'], 'E1012: Type mismatch; expected func(bool) but got func(bool, number)')
2289 v9.CheckDefFailure(['var Ref1: func(?bool)', 'Ref1 = g:FuncTwoArgNoRet'], 'E1012: Type mismatch; expected func(?bool) but got func(bool, number)')
2290 v9.CheckDefFailure(['var Ref1: func(...bool)', 'Ref1 = g:FuncTwoArgNoRet'], 'E1012: Type mismatch; expected func(...bool) but got func(bool, number)')
Bram Moolenaar08938ee2020-04-11 23:17:17 +02002291
Bram Moolenaar62aec932022-01-29 21:45:34 +00002292 v9.CheckDefFailure(['var RefWrong: func(string ,number)'], 'E1068:')
2293 v9.CheckDefFailure(['var RefWrong: func(string,number)'], 'E1069:')
2294 v9.CheckDefFailure(['var RefWrong: func(bool, bool, bool, bool, bool, bool, bool, bool, bool, bool, bool, bool, bool, bool, bool, bool, bool, bool, bool, bool)'], 'E1005:')
2295 v9.CheckDefFailure(['var RefWrong: func(bool):string'], 'E1069:')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002296enddef
2297
Bram Moolenaar89228602020-04-05 22:14:54 +02002298def Test_func_return_type()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002299 var nr: number
Bram Moolenaar62aec932022-01-29 21:45:34 +00002300 nr = g:FuncNoArgRetNumber()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002301 nr->assert_equal(1234)
Bram Moolenaar89228602020-04-05 22:14:54 +02002302
2303 nr = FuncOneArgRetAny(122)
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002304 nr->assert_equal(122)
Bram Moolenaar89228602020-04-05 22:14:54 +02002305
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002306 var str: string
Bram Moolenaar89228602020-04-05 22:14:54 +02002307 str = FuncOneArgRetAny('yes')
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002308 str->assert_equal('yes')
Bram Moolenaar89228602020-04-05 22:14:54 +02002309
Bram Moolenaar62aec932022-01-29 21:45:34 +00002310 v9.CheckDefFailure(['var str: string', 'str = g:FuncNoArgRetNumber()'], 'E1012: Type mismatch; expected string but got number')
Bram Moolenaar89228602020-04-05 22:14:54 +02002311enddef
2312
Bram Moolenaar6abd3dc2020-10-04 14:17:32 +02002313def Test_func_common_type()
2314 def FuncOne(n: number): number
2315 return n
2316 enddef
2317 def FuncTwo(s: string): number
2318 return len(s)
2319 enddef
2320 def FuncThree(n: number, s: string): number
2321 return n + len(s)
2322 enddef
2323 var list = [FuncOne, FuncTwo, FuncThree]
2324 assert_equal(8, list[0](8))
2325 assert_equal(4, list[1]('word'))
2326 assert_equal(7, list[2](3, 'word'))
2327enddef
2328
Bram Moolenaar62aec932022-01-29 21:45:34 +00002329def s:MultiLine(
Bram Moolenaar5e774c72020-04-12 21:53:00 +02002330 arg1: string,
2331 arg2 = 1234,
2332 ...rest: list<string>
2333 ): string
2334 return arg1 .. arg2 .. join(rest, '-')
2335enddef
2336
Bram Moolenaar2c330432020-04-13 14:41:35 +02002337def MultiLineComment(
2338 arg1: string, # comment
2339 arg2 = 1234, # comment
2340 ...rest: list<string> # comment
2341 ): string # comment
2342 return arg1 .. arg2 .. join(rest, '-')
2343enddef
2344
Bram Moolenaar5e774c72020-04-12 21:53:00 +02002345def Test_multiline()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002346 MultiLine('text')->assert_equal('text1234')
2347 MultiLine('text', 777)->assert_equal('text777')
2348 MultiLine('text', 777, 'one')->assert_equal('text777one')
2349 MultiLine('text', 777, 'one', 'two')->assert_equal('text777one-two')
Bram Moolenaar5e774c72020-04-12 21:53:00 +02002350enddef
2351
Bram Moolenaar23e03252020-04-12 22:22:31 +02002352func Test_multiline_not_vim9()
Bram Moolenaar62aec932022-01-29 21:45:34 +00002353 call s:MultiLine('text')->assert_equal('text1234')
2354 call s:MultiLine('text', 777)->assert_equal('text777')
2355 call s:MultiLine('text', 777, 'one')->assert_equal('text777one')
2356 call s:MultiLine('text', 777, 'one', 'two')->assert_equal('text777one-two')
Bram Moolenaar23e03252020-04-12 22:22:31 +02002357endfunc
2358
Bram Moolenaar5e774c72020-04-12 21:53:00 +02002359
Bram Moolenaaree4e0c12020-04-06 21:35:05 +02002360" When using CheckScriptFailure() for the below test, E1010 is generated instead
2361" of E1056.
2362func Test_E1056_1059()
2363 let caught_1056 = 0
2364 try
2365 def F():
2366 return 1
2367 enddef
2368 catch /E1056:/
2369 let caught_1056 = 1
2370 endtry
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002371 eval caught_1056->assert_equal(1)
Bram Moolenaaree4e0c12020-04-06 21:35:05 +02002372
2373 let caught_1059 = 0
2374 try
2375 def F5(items : list)
2376 echo 'a'
2377 enddef
2378 catch /E1059:/
2379 let caught_1059 = 1
2380 endtry
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002381 eval caught_1059->assert_equal(1)
Bram Moolenaaree4e0c12020-04-06 21:35:05 +02002382endfunc
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002383
Bram Moolenaar015f4262020-05-05 21:25:22 +02002384func DelMe()
2385 echo 'DelMe'
2386endfunc
2387
Bram Moolenaarbf8feb52020-08-08 14:26:31 +02002388def Test_error_reporting()
2389 # comment lines at the start of the function
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002390 var lines =<< trim END
Bram Moolenaarbf8feb52020-08-08 14:26:31 +02002391 " comment
2392 def Func()
2393 # comment
2394 # comment
2395 invalid
2396 enddef
2397 defcompile
2398 END
Bram Moolenaar08052222020-09-14 17:04:31 +02002399 writefile(lines, 'Xdef')
Bram Moolenaarbf8feb52020-08-08 14:26:31 +02002400 try
2401 source Xdef
Bram Moolenaar7517ffd2020-08-14 18:35:07 +02002402 assert_report('should have failed')
Bram Moolenaarbf8feb52020-08-08 14:26:31 +02002403 catch /E476:/
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002404 v:exception->assert_match('Invalid command: invalid')
2405 v:throwpoint->assert_match(', line 3$')
Bram Moolenaarbf8feb52020-08-08 14:26:31 +02002406 endtry
Bram Moolenaar2d870f82020-12-05 13:41:01 +01002407 delfunc! g:Func
Bram Moolenaarbf8feb52020-08-08 14:26:31 +02002408
2409 # comment lines after the start of the function
2410 lines =<< trim END
2411 " comment
2412 def Func()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002413 var x = 1234
Bram Moolenaarbf8feb52020-08-08 14:26:31 +02002414 # comment
2415 # comment
2416 invalid
2417 enddef
2418 defcompile
2419 END
Bram Moolenaar08052222020-09-14 17:04:31 +02002420 writefile(lines, 'Xdef')
Bram Moolenaarbf8feb52020-08-08 14:26:31 +02002421 try
2422 source Xdef
Bram Moolenaar7517ffd2020-08-14 18:35:07 +02002423 assert_report('should have failed')
Bram Moolenaarbf8feb52020-08-08 14:26:31 +02002424 catch /E476:/
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002425 v:exception->assert_match('Invalid command: invalid')
2426 v:throwpoint->assert_match(', line 4$')
Bram Moolenaarbf8feb52020-08-08 14:26:31 +02002427 endtry
Bram Moolenaar2d870f82020-12-05 13:41:01 +01002428 delfunc! g:Func
Bram Moolenaarbf8feb52020-08-08 14:26:31 +02002429
Bram Moolenaar7517ffd2020-08-14 18:35:07 +02002430 lines =<< trim END
2431 vim9script
2432 def Func()
Bram Moolenaare0de1712020-12-02 17:36:54 +01002433 var db = {foo: 1, bar: 2}
Bram Moolenaar7517ffd2020-08-14 18:35:07 +02002434 # comment
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002435 var x = db.asdf
Bram Moolenaar7517ffd2020-08-14 18:35:07 +02002436 enddef
2437 defcompile
2438 Func()
2439 END
Bram Moolenaar08052222020-09-14 17:04:31 +02002440 writefile(lines, 'Xdef')
Bram Moolenaar7517ffd2020-08-14 18:35:07 +02002441 try
2442 source Xdef
2443 assert_report('should have failed')
2444 catch /E716:/
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002445 v:throwpoint->assert_match('_Func, line 3$')
Bram Moolenaar7517ffd2020-08-14 18:35:07 +02002446 endtry
Bram Moolenaar2d870f82020-12-05 13:41:01 +01002447 delfunc! g:Func
Bram Moolenaar7517ffd2020-08-14 18:35:07 +02002448
Bram Moolenaar08052222020-09-14 17:04:31 +02002449 delete('Xdef')
Bram Moolenaarbf8feb52020-08-08 14:26:31 +02002450enddef
2451
Bram Moolenaar015f4262020-05-05 21:25:22 +02002452def Test_deleted_function()
Bram Moolenaar62aec932022-01-29 21:45:34 +00002453 v9.CheckDefExecFailure([
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002454 'var RefMe: func = function("g:DelMe")',
Bram Moolenaar015f4262020-05-05 21:25:22 +02002455 'delfunc g:DelMe',
2456 'echo RefMe()'], 'E117:')
2457enddef
2458
2459def Test_unknown_function()
Bram Moolenaar62aec932022-01-29 21:45:34 +00002460 v9.CheckDefExecFailure([
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002461 'var Ref: func = function("NotExist")',
Bram Moolenaar9b7bf9e2020-07-11 22:14:59 +02002462 'delfunc g:NotExist'], 'E700:')
Bram Moolenaar015f4262020-05-05 21:25:22 +02002463enddef
2464
Bram Moolenaar62aec932022-01-29 21:45:34 +00002465def s:RefFunc(Ref: func(any): any): string
Bram Moolenaarc8cd2b32020-05-01 19:29:08 +02002466 return Ref('more')
2467enddef
2468
2469def Test_closure_simple()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002470 var local = 'some '
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01002471 RefFunc((s) => local .. s)->assert_equal('some more')
Bram Moolenaarc8cd2b32020-05-01 19:29:08 +02002472enddef
2473
Bram Moolenaar62aec932022-01-29 21:45:34 +00002474def s:MakeRef()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002475 var local = 'some '
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01002476 g:Ref = (s) => local .. s
Bram Moolenaarbf67ea12020-05-02 17:52:42 +02002477enddef
2478
2479def Test_closure_ref_after_return()
2480 MakeRef()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002481 g:Ref('thing')->assert_equal('some thing')
Bram Moolenaarbf67ea12020-05-02 17:52:42 +02002482 unlet g:Ref
2483enddef
2484
Bram Moolenaar62aec932022-01-29 21:45:34 +00002485def s:MakeTwoRefs()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002486 var local = ['some']
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01002487 g:Extend = (s) => local->add(s)
2488 g:Read = () => local
Bram Moolenaar5adc55c2020-05-02 23:12:58 +02002489enddef
2490
2491def Test_closure_two_refs()
2492 MakeTwoRefs()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002493 join(g:Read(), ' ')->assert_equal('some')
Bram Moolenaar5adc55c2020-05-02 23:12:58 +02002494 g:Extend('more')
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002495 join(g:Read(), ' ')->assert_equal('some more')
Bram Moolenaar5adc55c2020-05-02 23:12:58 +02002496 g:Extend('even')
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002497 join(g:Read(), ' ')->assert_equal('some more even')
Bram Moolenaar5adc55c2020-05-02 23:12:58 +02002498
2499 unlet g:Extend
2500 unlet g:Read
2501enddef
2502
Bram Moolenaar62aec932022-01-29 21:45:34 +00002503def s:ReadRef(Ref: func(): list<string>): string
Bram Moolenaar5adc55c2020-05-02 23:12:58 +02002504 return join(Ref(), ' ')
2505enddef
2506
Bram Moolenaar62aec932022-01-29 21:45:34 +00002507def s:ExtendRef(Ref: func(string): list<string>, add: string)
Bram Moolenaar5adc55c2020-05-02 23:12:58 +02002508 Ref(add)
2509enddef
2510
2511def Test_closure_two_indirect_refs()
Bram Moolenaarf7779c62020-05-03 15:38:16 +02002512 MakeTwoRefs()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002513 ReadRef(g:Read)->assert_equal('some')
Bram Moolenaar5adc55c2020-05-02 23:12:58 +02002514 ExtendRef(g:Extend, 'more')
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002515 ReadRef(g:Read)->assert_equal('some more')
Bram Moolenaar5adc55c2020-05-02 23:12:58 +02002516 ExtendRef(g:Extend, 'even')
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002517 ReadRef(g:Read)->assert_equal('some more even')
Bram Moolenaar5adc55c2020-05-02 23:12:58 +02002518
2519 unlet g:Extend
2520 unlet g:Read
2521enddef
Bram Moolenaarbf67ea12020-05-02 17:52:42 +02002522
Bram Moolenaar62aec932022-01-29 21:45:34 +00002523def s:MakeArgRefs(theArg: string)
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002524 var local = 'loc_val'
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01002525 g:UseArg = (s) => theArg .. '/' .. local .. '/' .. s
Bram Moolenaar2fd4cd72020-05-03 22:30:49 +02002526enddef
2527
Bram Moolenaar62aec932022-01-29 21:45:34 +00002528def s:MakeArgRefsVarargs(theArg: string, ...rest: list<string>)
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002529 var local = 'the_loc'
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01002530 g:UseVararg = (s) => theArg .. '/' .. local .. '/' .. s .. '/' .. join(rest)
Bram Moolenaar2fd4cd72020-05-03 22:30:49 +02002531enddef
2532
2533def Test_closure_using_argument()
2534 MakeArgRefs('arg_val')
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002535 g:UseArg('call_val')->assert_equal('arg_val/loc_val/call_val')
Bram Moolenaar2fd4cd72020-05-03 22:30:49 +02002536
2537 MakeArgRefsVarargs('arg_val', 'one', 'two')
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002538 g:UseVararg('call_val')->assert_equal('arg_val/the_loc/call_val/one two')
Bram Moolenaar2fd4cd72020-05-03 22:30:49 +02002539
2540 unlet g:UseArg
2541 unlet g:UseVararg
Bram Moolenaar44ec21c2021-02-12 21:50:57 +01002542
2543 var lines =<< trim END
2544 vim9script
2545 def Test(Fun: func(number): number): list<number>
2546 return map([1, 2, 3], (_, i) => Fun(i))
2547 enddef
2548 def Inc(nr: number): number
2549 return nr + 2
2550 enddef
2551 assert_equal([3, 4, 5], Test(Inc))
2552 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002553 v9.CheckScriptSuccess(lines)
Bram Moolenaar2fd4cd72020-05-03 22:30:49 +02002554enddef
2555
Bram Moolenaar62aec932022-01-29 21:45:34 +00002556def s:MakeGetAndAppendRefs()
Bram Moolenaar85d5e2b2020-10-10 14:13:01 +02002557 var local = 'a'
2558
2559 def Append(arg: string)
2560 local ..= arg
2561 enddef
2562 g:Append = Append
2563
2564 def Get(): string
2565 return local
2566 enddef
2567 g:Get = Get
2568enddef
2569
2570def Test_closure_append_get()
2571 MakeGetAndAppendRefs()
2572 g:Get()->assert_equal('a')
2573 g:Append('-b')
2574 g:Get()->assert_equal('a-b')
2575 g:Append('-c')
2576 g:Get()->assert_equal('a-b-c')
2577
2578 unlet g:Append
2579 unlet g:Get
2580enddef
Bram Moolenaarb68b3462020-05-06 21:06:30 +02002581
Bram Moolenaar04b12692020-05-04 23:24:44 +02002582def Test_nested_closure()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002583 var local = 'text'
Bram Moolenaar04b12692020-05-04 23:24:44 +02002584 def Closure(arg: string): string
2585 return local .. arg
2586 enddef
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002587 Closure('!!!')->assert_equal('text!!!')
Bram Moolenaar04b12692020-05-04 23:24:44 +02002588enddef
2589
Bram Moolenaar62aec932022-01-29 21:45:34 +00002590func s:GetResult(Ref)
Bram Moolenaar6f5b6df2020-05-16 21:20:12 +02002591 return a:Ref('some')
2592endfunc
2593
2594def Test_call_closure_not_compiled()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002595 var text = 'text'
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01002596 g:Ref = (s) => s .. text
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002597 GetResult(g:Ref)->assert_equal('sometext')
Bram Moolenaar6f5b6df2020-05-16 21:20:12 +02002598enddef
2599
Bram Moolenaar7cbfaa52020-09-18 21:25:32 +02002600def Test_double_closure_fails()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002601 var lines =<< trim END
Bram Moolenaar7cbfaa52020-09-18 21:25:32 +02002602 vim9script
2603 def Func()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002604 var name = 0
2605 for i in range(2)
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01002606 timer_start(0, () => name)
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002607 endfor
Bram Moolenaar7cbfaa52020-09-18 21:25:32 +02002608 enddef
2609 Func()
2610 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002611 v9.CheckScriptSuccess(lines)
Bram Moolenaar7cbfaa52020-09-18 21:25:32 +02002612enddef
2613
Bram Moolenaar85d5e2b2020-10-10 14:13:01 +02002614def Test_nested_closure_used()
2615 var lines =<< trim END
2616 vim9script
2617 def Func()
2618 var x = 'hello'
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01002619 var Closure = () => x
2620 g:Myclosure = () => Closure()
Bram Moolenaar85d5e2b2020-10-10 14:13:01 +02002621 enddef
2622 Func()
2623 assert_equal('hello', g:Myclosure())
2624 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002625 v9.CheckScriptSuccess(lines)
Bram Moolenaar85d5e2b2020-10-10 14:13:01 +02002626enddef
Bram Moolenaar0876c782020-10-07 19:08:04 +02002627
Bram Moolenaarc70bdab2020-09-26 19:59:38 +02002628def Test_nested_closure_fails()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002629 var lines =<< trim END
Bram Moolenaarc70bdab2020-09-26 19:59:38 +02002630 vim9script
2631 def FuncA()
2632 FuncB(0)
2633 enddef
2634 def FuncB(n: number): list<string>
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01002635 return map([0], (_, v) => n)
Bram Moolenaarc70bdab2020-09-26 19:59:38 +02002636 enddef
2637 FuncA()
2638 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002639 v9.CheckScriptFailure(lines, 'E1012:')
Bram Moolenaarc70bdab2020-09-26 19:59:38 +02002640enddef
2641
Bram Moolenaarf112f302020-12-20 17:47:52 +01002642def Test_global_closure()
2643 var lines =<< trim END
2644 vim9script
2645 def ReverseEveryNLines(n: number, line1: number, line2: number)
2646 var mods = 'sil keepj keepp lockm '
2647 var range = ':' .. line1 .. ',' .. line2
2648 def g:Offset(): number
2649 var offset = (line('.') - line1 + 1) % n
2650 return offset != 0 ? offset : n
2651 enddef
2652 exe mods .. range .. 'g/^/exe "m .-" .. g:Offset()'
2653 enddef
2654
2655 new
2656 repeat(['aaa', 'bbb', 'ccc'], 3)->setline(1)
2657 ReverseEveryNLines(3, 1, 9)
2658 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002659 v9.CheckScriptSuccess(lines)
Bram Moolenaarf112f302020-12-20 17:47:52 +01002660 var expected = repeat(['ccc', 'bbb', 'aaa'], 3)
2661 assert_equal(expected, getline(1, 9))
2662 bwipe!
2663enddef
2664
Bram Moolenaarcd45ed02020-12-22 17:35:54 +01002665def Test_global_closure_called_directly()
2666 var lines =<< trim END
2667 vim9script
2668 def Outer()
2669 var x = 1
2670 def g:Inner()
2671 var y = x
2672 x += 1
2673 assert_equal(1, y)
2674 enddef
2675 g:Inner()
2676 assert_equal(2, x)
2677 enddef
2678 Outer()
2679 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002680 v9.CheckScriptSuccess(lines)
Bram Moolenaarcd45ed02020-12-22 17:35:54 +01002681 delfunc g:Inner
2682enddef
2683
Bram Moolenaar69c76172021-12-02 16:38:52 +00002684def Test_closure_called_from_legacy()
2685 var lines =<< trim END
2686 vim9script
2687 def Func()
2688 var outer = 'foo'
2689 var F = () => {
2690 outer = 'bar'
2691 }
2692 execute printf('call %s()', string(F))
2693 enddef
2694 Func()
2695 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002696 v9.CheckScriptFailure(lines, 'E1248')
Bram Moolenaar69c76172021-12-02 16:38:52 +00002697enddef
2698
Bram Moolenaar34c54eb2020-11-25 19:15:19 +01002699def Test_failure_in_called_function()
2700 # this was using the frame index as the return value
2701 var lines =<< trim END
2702 vim9script
2703 au TerminalWinOpen * eval [][0]
2704 def PopupTerm(a: any)
2705 # make sure typvals on stack are string
2706 ['a', 'b', 'c', 'd', 'e', 'f', 'g']->join()
2707 FireEvent()
2708 enddef
2709 def FireEvent()
2710 do TerminalWinOpen
2711 enddef
2712 # use try/catch to make eval fail
2713 try
2714 call PopupTerm(0)
2715 catch
2716 endtry
2717 au! TerminalWinOpen
2718 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002719 v9.CheckScriptSuccess(lines)
Bram Moolenaar34c54eb2020-11-25 19:15:19 +01002720enddef
2721
Bram Moolenaar5366e1a2020-10-01 13:01:34 +02002722def Test_nested_lambda()
2723 var lines =<< trim END
2724 vim9script
2725 def Func()
2726 var x = 4
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01002727 var Lambda1 = () => 7
2728 var Lambda2 = () => [Lambda1(), x]
Bram Moolenaar5366e1a2020-10-01 13:01:34 +02002729 var res = Lambda2()
2730 assert_equal([7, 4], res)
2731 enddef
2732 Func()
2733 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002734 v9.CheckScriptSuccess(lines)
Bram Moolenaar5366e1a2020-10-01 13:01:34 +02002735enddef
2736
Bram Moolenaarc04f2a42021-06-09 19:30:03 +02002737def Test_double_nested_lambda()
2738 var lines =<< trim END
2739 vim9script
2740 def F(head: string): func(string): func(string): string
2741 return (sep: string): func(string): string => ((tail: string): string => {
2742 return head .. sep .. tail
2743 })
2744 enddef
2745 assert_equal('hello-there', F('hello')('-')('there'))
2746 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002747 v9.CheckScriptSuccess(lines)
Bram Moolenaarc04f2a42021-06-09 19:30:03 +02002748enddef
2749
Bram Moolenaar074f84c2021-05-18 11:47:44 +02002750def Test_nested_inline_lambda()
Bram Moolenaar074f84c2021-05-18 11:47:44 +02002751 var lines =<< trim END
2752 vim9script
2753 def F(text: string): func(string): func(string): string
2754 return (arg: string): func(string): string => ((sep: string): string => {
Bram Moolenaar23e2e112021-08-03 21:16:18 +02002755 return sep .. arg .. text
Bram Moolenaar074f84c2021-05-18 11:47:44 +02002756 })
2757 enddef
Bram Moolenaar23e2e112021-08-03 21:16:18 +02002758 assert_equal('--there++', F('++')('there')('--'))
Bram Moolenaar074f84c2021-05-18 11:47:44 +02002759 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002760 v9.CheckScriptSuccess(lines)
Bram Moolenaar5245beb2021-07-15 22:03:50 +02002761
2762 lines =<< trim END
2763 vim9script
2764 echo range(4)->mapnew((_, v) => {
2765 return range(v) ->mapnew((_, s) => {
2766 return string(s)
2767 })
2768 })
2769 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002770 v9.CheckScriptSuccess(lines)
Bram Moolenaarc6ba2f92021-07-18 13:42:29 +02002771
2772 lines =<< trim END
2773 vim9script
2774
2775 def s:func()
2776 range(10)
2777 ->mapnew((_, _) => ({
2778 key: range(10)->mapnew((_, _) => {
2779 return ' '
2780 }),
2781 }))
2782 enddef
2783
2784 defcomp
2785 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002786 v9.CheckScriptSuccess(lines)
Bram Moolenaar074f84c2021-05-18 11:47:44 +02002787enddef
2788
Bram Moolenaar52bf81c2020-11-17 18:50:44 +01002789def Shadowed(): list<number>
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01002790 var FuncList: list<func: number> = [() => 42]
Bram Moolenaar75ab91f2021-01-10 22:42:50 +01002791 return FuncList->mapnew((_, Shadowed) => Shadowed())
Bram Moolenaar52bf81c2020-11-17 18:50:44 +01002792enddef
2793
2794def Test_lambda_arg_shadows_func()
Bram Moolenaar62aec932022-01-29 21:45:34 +00002795 assert_equal([42], g:Shadowed())
Bram Moolenaar52bf81c2020-11-17 18:50:44 +01002796enddef
2797
Bram Moolenaar62aec932022-01-29 21:45:34 +00002798def s:Line_continuation_in_def(dir: string = ''): string
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002799 var path: string = empty(dir)
2800 \ ? 'empty'
2801 \ : 'full'
2802 return path
Bram Moolenaaracd4c5e2020-06-22 19:39:03 +02002803enddef
2804
2805def Test_line_continuation_in_def()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002806 Line_continuation_in_def('.')->assert_equal('full')
Bram Moolenaaracd4c5e2020-06-22 19:39:03 +02002807enddef
2808
Bram Moolenaar2ea95b62020-11-19 21:47:56 +01002809def Test_script_var_in_lambda()
2810 var lines =<< trim END
2811 vim9script
2812 var script = 'test'
Bram Moolenaarbb8a7ce2021-04-10 20:10:26 +02002813 assert_equal(['test'], map(['one'], (_, _) => script))
Bram Moolenaar2ea95b62020-11-19 21:47:56 +01002814 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002815 v9.CheckScriptSuccess(lines)
Bram Moolenaar2ea95b62020-11-19 21:47:56 +01002816enddef
2817
Bram Moolenaar62aec932022-01-29 21:45:34 +00002818def s:Line_continuation_in_lambda(): list<string>
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002819 var x = range(97, 100)
Bram Moolenaar75ab91f2021-01-10 22:42:50 +01002820 ->mapnew((_, v) => nr2char(v)
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01002821 ->toupper())
Bram Moolenaar7a4b8982020-07-08 17:36:21 +02002822 ->reverse()
2823 return x
2824enddef
2825
2826def Test_line_continuation_in_lambda()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002827 Line_continuation_in_lambda()->assert_equal(['D', 'C', 'B', 'A'])
Bram Moolenaarf898f7c2021-01-16 18:09:52 +01002828
2829 var lines =<< trim END
2830 vim9script
2831 var res = [{n: 1, m: 2, s: 'xxx'}]
2832 ->mapnew((_, v: dict<any>): string => printf('%d:%d:%s',
2833 v.n,
2834 v.m,
2835 substitute(v.s, '.*', 'yyy', '')
2836 ))
2837 assert_equal(['1:2:yyy'], res)
2838 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002839 v9.CheckScriptSuccess(lines)
Bram Moolenaar7a4b8982020-07-08 17:36:21 +02002840enddef
2841
Bram Moolenaarb6571982021-01-08 22:24:19 +01002842def Test_list_lambda()
2843 timer_start(1000, (_) => 0)
2844 var body = execute(timer_info()[0].callback
2845 ->string()
2846 ->substitute("('", ' ', '')
2847 ->substitute("')", '', '')
2848 ->substitute('function\zs', ' ', ''))
Bram Moolenaar767034c2021-04-09 17:24:52 +02002849 assert_match('def <lambda>\d\+(_: any): number\n1 return 0\n enddef', body)
Bram Moolenaarb6571982021-01-08 22:24:19 +01002850enddef
2851
Bram Moolenaar3c77b6a2021-07-25 18:07:00 +02002852def Test_lambda_block_variable()
Bram Moolenaar88421d62021-07-24 14:14:52 +02002853 var lines =<< trim END
2854 vim9script
2855 var flist: list<func>
2856 for i in range(10)
2857 var inloop = i
2858 flist[i] = () => inloop
2859 endfor
2860 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002861 v9.CheckScriptSuccess(lines)
Bram Moolenaar88421d62021-07-24 14:14:52 +02002862
2863 lines =<< trim END
2864 vim9script
2865 if true
2866 var outloop = 5
2867 var flist: list<func>
2868 for i in range(10)
2869 flist[i] = () => outloop
2870 endfor
2871 endif
2872 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002873 v9.CheckScriptSuccess(lines)
Bram Moolenaar88421d62021-07-24 14:14:52 +02002874
2875 lines =<< trim END
2876 vim9script
2877 if true
2878 var outloop = 5
2879 endif
2880 var flist: list<func>
2881 for i in range(10)
2882 flist[i] = () => outloop
2883 endfor
2884 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002885 v9.CheckScriptFailure(lines, 'E1001: Variable not found: outloop', 1)
Bram Moolenaar3c77b6a2021-07-25 18:07:00 +02002886
2887 lines =<< trim END
2888 vim9script
2889 for i in range(10)
2890 var Ref = () => 0
2891 endfor
2892 assert_equal(0, ((i) => 0)(0))
2893 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002894 v9.CheckScriptSuccess(lines)
Bram Moolenaar88421d62021-07-24 14:14:52 +02002895enddef
2896
Bram Moolenaar96cf4ba2021-04-24 14:15:41 +02002897def Test_legacy_lambda()
2898 legacy echo {x -> 'hello ' .. x}('foo')
Bram Moolenaardc4c2302021-04-25 13:54:42 +02002899
Bram Moolenaar96cf4ba2021-04-24 14:15:41 +02002900 var lines =<< trim END
2901 echo {x -> 'hello ' .. x}('foo')
2902 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002903 v9.CheckDefAndScriptFailure(lines, 'E720:')
Bram Moolenaardc4c2302021-04-25 13:54:42 +02002904
2905 lines =<< trim END
2906 vim9script
2907 def Func()
2908 echo (() => 'no error')()
2909 enddef
2910 legacy call s:Func()
2911 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002912 v9.CheckScriptSuccess(lines)
Bram Moolenaar96cf4ba2021-04-24 14:15:41 +02002913enddef
2914
Bram Moolenaarce024c32021-06-26 13:00:49 +02002915def Test_legacy()
2916 var lines =<< trim END
2917 vim9script
2918 func g:LegacyFunction()
2919 let g:legacyvar = 1
2920 endfunc
2921 def Testit()
2922 legacy call g:LegacyFunction()
2923 enddef
2924 Testit()
2925 assert_equal(1, g:legacyvar)
2926 unlet g:legacyvar
2927 delfunc g:LegacyFunction
2928 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002929 v9.CheckScriptSuccess(lines)
Bram Moolenaarce024c32021-06-26 13:00:49 +02002930enddef
2931
Bram Moolenaarc3cb1c92021-06-02 16:47:53 +02002932def Test_legacy_errors()
2933 for cmd in ['if', 'elseif', 'else', 'endif',
2934 'for', 'endfor', 'continue', 'break',
2935 'while', 'endwhile',
2936 'try', 'catch', 'finally', 'endtry']
Bram Moolenaar62aec932022-01-29 21:45:34 +00002937 v9.CheckDefFailure(['legacy ' .. cmd .. ' expr'], 'E1189:')
Bram Moolenaarc3cb1c92021-06-02 16:47:53 +02002938 endfor
2939enddef
2940
Bram Moolenaarb1b6f4d2021-09-13 18:25:54 +02002941def Test_call_legacy_with_dict()
2942 var lines =<< trim END
2943 vim9script
2944 func Legacy() dict
2945 let g:result = self.value
2946 endfunc
2947 def TestDirect()
2948 var d = {value: 'yes', func: Legacy}
2949 d.func()
2950 enddef
2951 TestDirect()
2952 assert_equal('yes', g:result)
2953 unlet g:result
2954
2955 def TestIndirect()
2956 var d = {value: 'foo', func: Legacy}
2957 var Fi = d.func
2958 Fi()
2959 enddef
2960 TestIndirect()
2961 assert_equal('foo', g:result)
2962 unlet g:result
2963
2964 var d = {value: 'bar', func: Legacy}
2965 d.func()
2966 assert_equal('bar', g:result)
2967 unlet g:result
2968 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002969 v9.CheckScriptSuccess(lines)
Bram Moolenaarb1b6f4d2021-09-13 18:25:54 +02002970enddef
2971
Bram Moolenaar62aec932022-01-29 21:45:34 +00002972def s:DoFilterThis(a: string): list<string>
Bram Moolenaarab360522021-01-10 14:02:28 +01002973 # closure nested inside another closure using argument
2974 var Filter = (l) => filter(l, (_, v) => stridx(v, a) == 0)
2975 return ['x', 'y', 'a', 'x2', 'c']->Filter()
2976enddef
2977
2978def Test_nested_closure_using_argument()
2979 assert_equal(['x', 'x2'], DoFilterThis('x'))
2980enddef
2981
Bram Moolenaar0186e582021-01-10 18:33:11 +01002982def Test_triple_nested_closure()
2983 var what = 'x'
2984 var Match = (val: string, cmp: string): bool => stridx(val, cmp) == 0
2985 var Filter = (l) => filter(l, (_, v) => Match(v, what))
2986 assert_equal(['x', 'x2'], ['x', 'y', 'a', 'x2', 'c']->Filter())
2987enddef
2988
Bram Moolenaar8f510af2020-07-05 18:48:23 +02002989func Test_silent_echo()
Bram Moolenaar47e7d702020-07-05 18:18:42 +02002990 CheckScreendump
Bram Moolenaar3b309f12021-12-13 18:19:55 +00002991 call Run_Test_silent_echo()
2992endfunc
Bram Moolenaar47e7d702020-07-05 18:18:42 +02002993
Bram Moolenaar3b309f12021-12-13 18:19:55 +00002994def Run_Test_silent_echo()
2995 var lines =<< trim END
Bram Moolenaar47e7d702020-07-05 18:18:42 +02002996 vim9script
2997 def EchoNothing()
2998 silent echo ''
2999 enddef
3000 defcompile
3001 END
Bram Moolenaar3b309f12021-12-13 18:19:55 +00003002 writefile(lines, 'XTest_silent_echo')
Bram Moolenaar47e7d702020-07-05 18:18:42 +02003003
Bram Moolenaar3b309f12021-12-13 18:19:55 +00003004 # Check that the balloon shows up after a mouse move
Bram Moolenaar62aec932022-01-29 21:45:34 +00003005 var buf = g:RunVimInTerminal('-S XTest_silent_echo', {'rows': 6})
Bram Moolenaar3b309f12021-12-13 18:19:55 +00003006 term_sendkeys(buf, ":abc")
Bram Moolenaar62aec932022-01-29 21:45:34 +00003007 g:VerifyScreenDump(buf, 'Test_vim9_silent_echo', {})
Bram Moolenaar47e7d702020-07-05 18:18:42 +02003008
Bram Moolenaar3b309f12021-12-13 18:19:55 +00003009 # clean up
Bram Moolenaar62aec932022-01-29 21:45:34 +00003010 g:StopVimInTerminal(buf)
Bram Moolenaar3b309f12021-12-13 18:19:55 +00003011 delete('XTest_silent_echo')
3012enddef
Bram Moolenaar47e7d702020-07-05 18:18:42 +02003013
Bram Moolenaar171fb922020-10-28 16:54:47 +01003014def SilentlyError()
3015 execute('silent! invalid')
3016 g:did_it = 'yes'
3017enddef
3018
Bram Moolenaar62aec932022-01-29 21:45:34 +00003019func s:UserError()
Bram Moolenaar28ee8922020-10-28 20:20:00 +01003020 silent! invalid
3021endfunc
3022
3023def SilentlyUserError()
3024 UserError()
3025 g:did_it = 'yes'
3026enddef
Bram Moolenaar171fb922020-10-28 16:54:47 +01003027
3028" This can't be a :def function, because the assert would not be reached.
Bram Moolenaar171fb922020-10-28 16:54:47 +01003029func Test_ignore_silent_error()
3030 let g:did_it = 'no'
3031 call SilentlyError()
3032 call assert_equal('yes', g:did_it)
3033
Bram Moolenaar28ee8922020-10-28 20:20:00 +01003034 let g:did_it = 'no'
3035 call SilentlyUserError()
3036 call assert_equal('yes', g:did_it)
Bram Moolenaar171fb922020-10-28 16:54:47 +01003037
3038 unlet g:did_it
3039endfunc
3040
Bram Moolenaarcd030c42020-10-30 21:49:40 +01003041def Test_ignore_silent_error_in_filter()
3042 var lines =<< trim END
3043 vim9script
3044 def Filter(winid: number, key: string): bool
3045 if key == 'o'
3046 silent! eval [][0]
3047 return true
3048 endif
3049 return popup_filter_menu(winid, key)
3050 enddef
3051
Bram Moolenaare0de1712020-12-02 17:36:54 +01003052 popup_create('popup', {filter: Filter})
Bram Moolenaarcd030c42020-10-30 21:49:40 +01003053 feedkeys("o\r", 'xnt')
3054 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003055 v9.CheckScriptSuccess(lines)
Bram Moolenaarcd030c42020-10-30 21:49:40 +01003056enddef
3057
Bram Moolenaar62aec932022-01-29 21:45:34 +00003058def s:Fibonacci(n: number): number
Bram Moolenaar4b9bd692020-09-05 21:57:53 +02003059 if n < 2
3060 return n
3061 else
3062 return Fibonacci(n - 1) + Fibonacci(n - 2)
3063 endif
3064enddef
3065
Bram Moolenaar985116a2020-07-12 17:31:09 +02003066def Test_recursive_call()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02003067 Fibonacci(20)->assert_equal(6765)
Bram Moolenaar985116a2020-07-12 17:31:09 +02003068enddef
3069
Bram Moolenaar62aec932022-01-29 21:45:34 +00003070def s:TreeWalk(dir: string): list<any>
Bram Moolenaar75ab91f2021-01-10 22:42:50 +01003071 return readdir(dir)->mapnew((_, val) =>
Bram Moolenaar08f7a412020-07-13 20:41:08 +02003072 fnamemodify(dir .. '/' .. val, ':p')->isdirectory()
Bram Moolenaar2bede172020-11-19 18:53:18 +01003073 ? {[val]: TreeWalk(dir .. '/' .. val)}
Bram Moolenaar08f7a412020-07-13 20:41:08 +02003074 : val
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01003075 )
Bram Moolenaar08f7a412020-07-13 20:41:08 +02003076enddef
3077
3078def Test_closure_in_map()
3079 mkdir('XclosureDir/tdir', 'p')
3080 writefile(['111'], 'XclosureDir/file1')
3081 writefile(['222'], 'XclosureDir/file2')
3082 writefile(['333'], 'XclosureDir/tdir/file3')
3083
Bram Moolenaare0de1712020-12-02 17:36:54 +01003084 TreeWalk('XclosureDir')->assert_equal(['file1', 'file2', {tdir: ['file3']}])
Bram Moolenaar08f7a412020-07-13 20:41:08 +02003085
3086 delete('XclosureDir', 'rf')
3087enddef
3088
Bram Moolenaar7b5d5442020-10-04 13:42:34 +02003089def Test_invalid_function_name()
3090 var lines =<< trim END
3091 vim9script
3092 def s: list<string>
3093 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003094 v9.CheckScriptFailure(lines, 'E129:')
Bram Moolenaar7b5d5442020-10-04 13:42:34 +02003095
3096 lines =<< trim END
3097 vim9script
3098 def g: list<string>
3099 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003100 v9.CheckScriptFailure(lines, 'E129:')
Bram Moolenaar7b5d5442020-10-04 13:42:34 +02003101
3102 lines =<< trim END
3103 vim9script
3104 def <SID>: list<string>
3105 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003106 v9.CheckScriptFailure(lines, 'E884:')
Bram Moolenaar7b5d5442020-10-04 13:42:34 +02003107
3108 lines =<< trim END
3109 vim9script
3110 def F list<string>
3111 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003112 v9.CheckScriptFailure(lines, 'E488:')
Bram Moolenaar7b5d5442020-10-04 13:42:34 +02003113enddef
3114
Bram Moolenaara90afb92020-07-15 22:38:56 +02003115def Test_partial_call()
Bram Moolenaarf78da4f2021-08-01 15:40:31 +02003116 var lines =<< trim END
3117 var Xsetlist: func
3118 Xsetlist = function('setloclist', [0])
3119 Xsetlist([], ' ', {title: 'test'})
3120 getloclist(0, {title: 1})->assert_equal({title: 'test'})
Bram Moolenaara90afb92020-07-15 22:38:56 +02003121
Bram Moolenaarf78da4f2021-08-01 15:40:31 +02003122 Xsetlist = function('setloclist', [0, [], ' '])
3123 Xsetlist({title: 'test'})
3124 getloclist(0, {title: 1})->assert_equal({title: 'test'})
Bram Moolenaara90afb92020-07-15 22:38:56 +02003125
Bram Moolenaarf78da4f2021-08-01 15:40:31 +02003126 Xsetlist = function('setqflist')
3127 Xsetlist([], ' ', {title: 'test'})
3128 getqflist({title: 1})->assert_equal({title: 'test'})
Bram Moolenaara90afb92020-07-15 22:38:56 +02003129
Bram Moolenaarf78da4f2021-08-01 15:40:31 +02003130 Xsetlist = function('setqflist', [[], ' '])
3131 Xsetlist({title: 'test'})
3132 getqflist({title: 1})->assert_equal({title: 'test'})
Bram Moolenaar6abd3dc2020-10-04 14:17:32 +02003133
Bram Moolenaarf78da4f2021-08-01 15:40:31 +02003134 var Len: func: number = function('len', ['word'])
3135 assert_equal(4, Len())
3136
3137 var RepeatFunc = function('repeat', ['o'])
3138 assert_equal('ooooo', RepeatFunc(5))
3139 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003140 v9.CheckDefAndScriptSuccess(lines)
Bram Moolenaarc66f6452021-08-19 21:08:30 +02003141
3142 lines =<< trim END
3143 vim9script
3144 def Foo(Parser: any)
3145 enddef
3146 var Expr: func(dict<any>): dict<any>
3147 const Call = Foo(Expr)
3148 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003149 v9.CheckScriptFailure(lines, 'E1235:')
Bram Moolenaara90afb92020-07-15 22:38:56 +02003150enddef
3151
Bram Moolenaarfe1bfc92022-02-06 13:55:03 +00003152" Using "idx" from a legacy global function does not work.
3153" This caused a crash when called from legacy context.
3154func Test_partial_call_fails()
3155 let lines =<< trim END
3156 vim9script
3157
3158 var l = ['a', 'b', 'c']
3159 def Iter(container: any): any
3160 var idx = -1
3161 var obj = {state: container}
Bram Moolenaarf681cfb2022-02-07 20:30:57 +00003162 def g:NextItem__(self: dict<any>): any
Bram Moolenaarfe1bfc92022-02-06 13:55:03 +00003163 ++idx
3164 return self.state[idx]
3165 enddef
Bram Moolenaarf681cfb2022-02-07 20:30:57 +00003166 obj.__next__ = function('g:NextItem__', [obj])
Bram Moolenaarfe1bfc92022-02-06 13:55:03 +00003167 return obj
3168 enddef
3169
3170 var it = Iter(l)
3171 echo it.__next__()
3172 END
3173 call writefile(lines, 'XpartialCall')
3174 try
3175 source XpartialCall
3176 catch /E1248:/
3177 endtry
3178 call delete('XpartialCall')
3179endfunc
3180
Bram Moolenaar2dd0a2c2020-08-08 15:10:27 +02003181def Test_cmd_modifier()
3182 tab echo '0'
Bram Moolenaar62aec932022-01-29 21:45:34 +00003183 v9.CheckDefFailure(['5tab echo 3'], 'E16:')
Bram Moolenaar2dd0a2c2020-08-08 15:10:27 +02003184enddef
3185
3186def Test_restore_modifiers()
3187 # check that when compiling a :def function command modifiers are not messed
3188 # up.
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02003189 var lines =<< trim END
Bram Moolenaar2dd0a2c2020-08-08 15:10:27 +02003190 vim9script
3191 set eventignore=
3192 autocmd QuickFixCmdPost * copen
3193 def AutocmdsDisabled()
Bram Moolenaarc3235272021-07-10 19:42:03 +02003194 eval 1 + 2
Bram Moolenaar2dd0a2c2020-08-08 15:10:27 +02003195 enddef
3196 func Func()
3197 noautocmd call s:AutocmdsDisabled()
3198 let g:ei_after = &eventignore
3199 endfunc
3200 Func()
3201 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003202 v9.CheckScriptSuccess(lines)
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02003203 g:ei_after->assert_equal('')
Bram Moolenaar2dd0a2c2020-08-08 15:10:27 +02003204enddef
3205
Bram Moolenaardfa3d552020-09-10 22:05:08 +02003206def StackTop()
Bram Moolenaarc3235272021-07-10 19:42:03 +02003207 eval 1 + 2
3208 eval 2 + 3
Bram Moolenaardfa3d552020-09-10 22:05:08 +02003209 # call not on fourth line
Bram Moolenaar62aec932022-01-29 21:45:34 +00003210 g:StackBot()
Bram Moolenaardfa3d552020-09-10 22:05:08 +02003211enddef
3212
3213def StackBot()
3214 # throw an error
3215 eval [][0]
3216enddef
3217
3218def Test_callstack_def()
3219 try
Bram Moolenaar62aec932022-01-29 21:45:34 +00003220 g:StackTop()
Bram Moolenaardfa3d552020-09-10 22:05:08 +02003221 catch
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02003222 v:throwpoint->assert_match('Test_callstack_def\[2\]..StackTop\[4\]..StackBot, line 2')
Bram Moolenaardfa3d552020-09-10 22:05:08 +02003223 endtry
3224enddef
3225
Bram Moolenaare8211a32020-10-09 22:04:29 +02003226" Re-using spot for variable used in block
3227def Test_block_scoped_var()
3228 var lines =<< trim END
3229 vim9script
3230 def Func()
3231 var x = ['a', 'b', 'c']
3232 if 1
3233 var y = 'x'
Bram Moolenaarbb8a7ce2021-04-10 20:10:26 +02003234 map(x, (_, _) => y)
Bram Moolenaare8211a32020-10-09 22:04:29 +02003235 endif
3236 var z = x
3237 assert_equal(['x', 'x', 'x'], z)
3238 enddef
3239 Func()
3240 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003241 v9.CheckScriptSuccess(lines)
Bram Moolenaare8211a32020-10-09 22:04:29 +02003242enddef
3243
Bram Moolenaareeece9e2020-11-20 19:26:48 +01003244def Test_reset_did_emsg()
3245 var lines =<< trim END
3246 @s = 'blah'
3247 au BufWinLeave * #
3248 def Func()
3249 var winid = popup_create('popup', {})
3250 exe '*s'
3251 popup_close(winid)
3252 enddef
3253 Func()
3254 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003255 v9.CheckScriptFailure(lines, 'E492:', 8)
Bram Moolenaar2d870f82020-12-05 13:41:01 +01003256 delfunc! g:Func
Bram Moolenaareeece9e2020-11-20 19:26:48 +01003257enddef
3258
Bram Moolenaar57f799e2020-12-12 20:42:19 +01003259def Test_did_emsg_reset()
3260 # executing an autocommand resets did_emsg, this should not result in a
3261 # builtin function considered failing
3262 var lines =<< trim END
3263 vim9script
3264 au BufWinLeave * #
3265 def Func()
Bram Moolenaar767034c2021-04-09 17:24:52 +02003266 popup_menu('', {callback: (a, b) => popup_create('', {})->popup_close()})
Bram Moolenaar57f799e2020-12-12 20:42:19 +01003267 eval [][0]
3268 enddef
3269 nno <F3> <cmd>call <sid>Func()<cr>
3270 feedkeys("\<F3>\e", 'xt')
3271 END
3272 writefile(lines, 'XemsgReset')
3273 assert_fails('so XemsgReset', ['E684:', 'E684:'], lines, 2)
3274 delete('XemsgReset')
3275 nunmap <F3>
3276 au! BufWinLeave
3277enddef
3278
Bram Moolenaar56602ba2020-12-05 21:22:08 +01003279def Test_abort_with_silent_call()
3280 var lines =<< trim END
3281 vim9script
3282 g:result = 'none'
3283 def Func()
3284 g:result += 3
3285 g:result = 'yes'
3286 enddef
3287 # error is silenced, but function aborts on error
3288 silent! Func()
3289 assert_equal('none', g:result)
3290 unlet g:result
3291 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003292 v9.CheckScriptSuccess(lines)
Bram Moolenaar56602ba2020-12-05 21:22:08 +01003293enddef
3294
Bram Moolenaarf665e972020-12-05 19:17:16 +01003295def Test_continues_with_silent_error()
3296 var lines =<< trim END
3297 vim9script
3298 g:result = 'none'
3299 def Func()
3300 silent! g:result += 3
3301 g:result = 'yes'
3302 enddef
3303 # error is silenced, function does not abort
3304 Func()
3305 assert_equal('yes', g:result)
3306 unlet g:result
3307 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003308 v9.CheckScriptSuccess(lines)
Bram Moolenaarf665e972020-12-05 19:17:16 +01003309enddef
3310
Bram Moolenaaraf0df472020-12-02 20:51:22 +01003311def Test_abort_even_with_silent()
3312 var lines =<< trim END
3313 vim9script
3314 g:result = 'none'
3315 def Func()
3316 eval {-> ''}() .. '' .. {}['X']
3317 g:result = 'yes'
3318 enddef
Bram Moolenaarf665e972020-12-05 19:17:16 +01003319 silent! Func()
Bram Moolenaaraf0df472020-12-02 20:51:22 +01003320 assert_equal('none', g:result)
Bram Moolenaar4029cab2020-12-05 18:13:27 +01003321 unlet g:result
3322 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003323 v9.CheckScriptSuccess(lines)
Bram Moolenaar4029cab2020-12-05 18:13:27 +01003324enddef
3325
Bram Moolenaarf665e972020-12-05 19:17:16 +01003326def Test_cmdmod_silent_restored()
3327 var lines =<< trim END
3328 vim9script
3329 def Func()
3330 g:result = 'none'
3331 silent! g:result += 3
3332 g:result = 'none'
3333 g:result += 3
3334 enddef
3335 Func()
3336 END
3337 # can't use CheckScriptFailure, it ignores the :silent!
3338 var fname = 'Xdefsilent'
3339 writefile(lines, fname)
3340 var caught = 'no'
3341 try
3342 exe 'source ' .. fname
3343 catch /E1030:/
3344 caught = 'yes'
3345 assert_match('Func, line 4', v:throwpoint)
3346 endtry
3347 assert_equal('yes', caught)
3348 delete(fname)
3349enddef
3350
Bram Moolenaar2fecb532021-03-24 22:00:56 +01003351def Test_cmdmod_silent_nested()
3352 var lines =<< trim END
3353 vim9script
3354 var result = ''
3355
3356 def Error()
3357 result ..= 'Eb'
3358 eval [][0]
3359 result ..= 'Ea'
3360 enddef
3361
3362 def Crash()
3363 result ..= 'Cb'
3364 sil! Error()
3365 result ..= 'Ca'
3366 enddef
3367
3368 Crash()
3369 assert_equal('CbEbEaCa', result)
3370 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003371 v9.CheckScriptSuccess(lines)
Bram Moolenaar2fecb532021-03-24 22:00:56 +01003372enddef
3373
Bram Moolenaar4029cab2020-12-05 18:13:27 +01003374def Test_dict_member_with_silent()
3375 var lines =<< trim END
3376 vim9script
3377 g:result = 'none'
3378 var d: dict<any>
3379 def Func()
3380 try
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01003381 g:result = map([], (_, v) => ({}[v]))->join() .. d['']
Bram Moolenaar4029cab2020-12-05 18:13:27 +01003382 catch
3383 endtry
3384 enddef
3385 silent! Func()
3386 assert_equal('0', g:result)
3387 unlet g:result
Bram Moolenaaraf0df472020-12-02 20:51:22 +01003388 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003389 v9.CheckScriptSuccess(lines)
Bram Moolenaaraf0df472020-12-02 20:51:22 +01003390enddef
3391
Bram Moolenaarf9041332021-01-21 19:41:16 +01003392def Test_skip_cmds_with_silent()
3393 var lines =<< trim END
3394 vim9script
3395
3396 def Func(b: bool)
3397 Crash()
3398 enddef
3399
3400 def Crash()
3401 sil! :/not found/d _
3402 sil! :/not found/put _
3403 enddef
3404
3405 Func(true)
3406 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003407 v9.CheckScriptSuccess(lines)
Bram Moolenaarf9041332021-01-21 19:41:16 +01003408enddef
3409
Bram Moolenaar5b3d1bb2020-12-22 12:20:08 +01003410def Test_opfunc()
Bram Moolenaar848fadd2022-01-30 15:28:30 +00003411 nnoremap <F3> <cmd>set opfunc=g:Opfunc<cr>g@
Bram Moolenaar5b3d1bb2020-12-22 12:20:08 +01003412 def g:Opfunc(_: any): string
3413 setline(1, 'ASDF')
3414 return ''
3415 enddef
3416 new
3417 setline(1, 'asdf')
3418 feedkeys("\<F3>$", 'x')
3419 assert_equal('ASDF', getline(1))
3420
3421 bwipe!
3422 nunmap <F3>
3423enddef
3424
Bram Moolenaar3b309f12021-12-13 18:19:55 +00003425func Test_opfunc_error()
3426 CheckScreendump
3427 call Run_Test_opfunc_error()
3428endfunc
3429
3430def Run_Test_opfunc_error()
3431 # test that the error from Opfunc() is displayed right away
3432 var lines =<< trim END
3433 vim9script
3434
3435 def Opfunc(type: string)
3436 try
3437 eval [][0]
3438 catch /nothing/ # error not caught
3439 endtry
3440 enddef
3441 &operatorfunc = Opfunc
3442 nnoremap <expr> l <SID>L()
3443 def L(): string
3444 return 'l'
3445 enddef
3446 'x'->repeat(10)->setline(1)
3447 feedkeys('g@l', 'n')
3448 feedkeys('llll')
3449 END
3450 call writefile(lines, 'XTest_opfunc_error')
3451
Bram Moolenaar62aec932022-01-29 21:45:34 +00003452 var buf = g:RunVimInTerminal('-S XTest_opfunc_error', {rows: 6, wait_for_ruler: 0})
3453 g:WaitForAssert(() => assert_match('Press ENTER', term_getline(buf, 6)))
3454 g:WaitForAssert(() => assert_match('E684: list index out of range: 0', term_getline(buf, 5)))
Bram Moolenaar3b309f12021-12-13 18:19:55 +00003455
3456 # clean up
Bram Moolenaar62aec932022-01-29 21:45:34 +00003457 g:StopVimInTerminal(buf)
Bram Moolenaar3b309f12021-12-13 18:19:55 +00003458 delete('XTest_opfunc_error')
3459enddef
3460
Bram Moolenaar077a4232020-12-22 18:33:27 +01003461" this was crashing on exit
3462def Test_nested_lambda_in_closure()
3463 var lines =<< trim END
3464 vim9script
Bram Moolenaar227c58a2021-04-28 20:40:44 +02003465 command WriteDone writefile(['Done'], 'XnestedDone')
Bram Moolenaar077a4232020-12-22 18:33:27 +01003466 def Outer()
3467 def g:Inner()
3468 echo map([1, 2, 3], {_, v -> v + 1})
3469 enddef
3470 g:Inner()
3471 enddef
3472 defcompile
Bram Moolenaar227c58a2021-04-28 20:40:44 +02003473 # not reached
Bram Moolenaar077a4232020-12-22 18:33:27 +01003474 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003475 if !g:RunVim([], lines, '--clean -c WriteDone -c quit')
Bram Moolenaar077a4232020-12-22 18:33:27 +01003476 return
3477 endif
3478 assert_equal(['Done'], readfile('XnestedDone'))
3479 delete('XnestedDone')
3480enddef
3481
Bram Moolenaar92368aa2022-02-07 17:50:39 +00003482def Test_nested_closure_funcref()
3483 var lines =<< trim END
3484 vim9script
3485 def Func()
3486 var n: number
3487 def Nested()
3488 ++n
3489 enddef
3490 Nested()
3491 g:result_one = n
3492 var Ref = function(Nested)
3493 Ref()
3494 g:result_two = n
3495 enddef
3496 Func()
3497 END
3498 v9.CheckScriptSuccess(lines)
3499 assert_equal(1, g:result_one)
3500 assert_equal(2, g:result_two)
3501 unlet g:result_one g:result_two
3502enddef
3503
Bram Moolenaar7aca5ca2022-02-07 19:56:43 +00003504def Test_nested_closure_in_dict()
3505 var lines =<< trim END
3506 vim9script
3507 def Func(): dict<any>
3508 var n: number
3509 def Inc(): number
3510 ++n
3511 return n
3512 enddef
3513 return {inc: function(Inc)}
3514 enddef
3515 disas Func
3516 var d = Func()
3517 assert_equal(1, d.inc())
3518 assert_equal(2, d.inc())
3519 END
3520 v9.CheckScriptSuccess(lines)
3521enddef
3522
Bram Moolenaar04947cc2021-03-06 19:26:46 +01003523def Test_check_func_arg_types()
3524 var lines =<< trim END
3525 vim9script
3526 def F1(x: string): string
3527 return x
3528 enddef
3529
3530 def F2(x: number): number
3531 return x + 1
3532 enddef
3533
3534 def G(g: func): dict<func>
3535 return {f: g}
3536 enddef
3537
3538 def H(d: dict<func>): string
3539 return d.f('a')
3540 enddef
3541 END
3542
Bram Moolenaar62aec932022-01-29 21:45:34 +00003543 v9.CheckScriptSuccess(lines + ['echo H(G(F1))'])
3544 v9.CheckScriptFailure(lines + ['echo H(G(F2))'], 'E1013:')
Bram Moolenaar04947cc2021-03-06 19:26:46 +01003545enddef
3546
Bram Moolenaar6e48b842021-08-10 22:52:02 +02003547def Test_list_any_type_checked()
3548 var lines =<< trim END
3549 vim9script
3550 def Foo()
3551 --decl--
3552 Bar(l)
3553 enddef
3554 def Bar(ll: list<dict<any>>)
3555 enddef
3556 Foo()
3557 END
Bram Moolenaar2d3ac2e2022-02-03 12:34:05 +00003558 # "any" could be "dict<any>", thus OK
Bram Moolenaar6e48b842021-08-10 22:52:02 +02003559 lines[2] = 'var l: list<any>'
Bram Moolenaar2d3ac2e2022-02-03 12:34:05 +00003560 v9.CheckScriptSuccess(lines)
Bram Moolenaar6e48b842021-08-10 22:52:02 +02003561 lines[2] = 'var l: list<any> = []'
Bram Moolenaar2d3ac2e2022-02-03 12:34:05 +00003562 v9.CheckScriptSuccess(lines)
Bram Moolenaar6e48b842021-08-10 22:52:02 +02003563
3564 lines[2] = 'var l: list<any> = [11]'
Bram Moolenaar62aec932022-01-29 21:45:34 +00003565 v9.CheckScriptFailure(lines, 'E1013: Argument 1: type mismatch, expected list<dict<any>> but got list<number>', 2)
Bram Moolenaar6e48b842021-08-10 22:52:02 +02003566enddef
3567
Bram Moolenaar701cc6c2021-04-10 13:33:48 +02003568def Test_compile_error()
3569 var lines =<< trim END
3570 def g:Broken()
3571 echo 'a' + {}
3572 enddef
3573 call g:Broken()
3574 END
3575 # First call: compilation error
Bram Moolenaar62aec932022-01-29 21:45:34 +00003576 v9.CheckScriptFailure(lines, 'E1051: Wrong argument type for +')
Bram Moolenaar701cc6c2021-04-10 13:33:48 +02003577
3578 # Second call won't try compiling again
3579 assert_fails('call g:Broken()', 'E1091: Function is not compiled: Broken')
Bram Moolenaar599410c2021-04-10 14:03:43 +02003580 delfunc g:Broken
3581
3582 # No error when compiling with :silent!
3583 lines =<< trim END
3584 def g:Broken()
3585 echo 'a' + []
3586 enddef
3587 silent! defcompile
3588 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003589 v9.CheckScriptSuccess(lines)
Bram Moolenaar599410c2021-04-10 14:03:43 +02003590
3591 # Calling the function won't try compiling again
3592 assert_fails('call g:Broken()', 'E1091: Function is not compiled: Broken')
3593 delfunc g:Broken
Bram Moolenaar701cc6c2021-04-10 13:33:48 +02003594enddef
3595
Bram Moolenaar962c43b2021-04-10 17:18:09 +02003596def Test_ignored_argument()
3597 var lines =<< trim END
3598 vim9script
3599 def Ignore(_, _): string
3600 return 'yes'
3601 enddef
3602 assert_equal('yes', Ignore(1, 2))
3603
3604 func Ok(_)
3605 return a:_
3606 endfunc
3607 assert_equal('ok', Ok('ok'))
3608
3609 func Oktoo()
3610 let _ = 'too'
3611 return _
3612 endfunc
3613 assert_equal('too', Oktoo())
Bram Moolenaarda479c72021-04-10 21:01:38 +02003614
3615 assert_equal([[1], [2], [3]], range(3)->mapnew((_, v) => [v]->map((_, w) => w + 1)))
Bram Moolenaar962c43b2021-04-10 17:18:09 +02003616 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003617 v9.CheckScriptSuccess(lines)
Bram Moolenaar962c43b2021-04-10 17:18:09 +02003618
3619 lines =<< trim END
3620 def Ignore(_: string): string
3621 return _
3622 enddef
3623 defcompile
3624 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003625 v9.CheckScriptFailure(lines, 'E1181:', 1)
Bram Moolenaar962c43b2021-04-10 17:18:09 +02003626
3627 lines =<< trim END
3628 var _ = 1
3629 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003630 v9.CheckDefAndScriptFailure(lines, 'E1181:', 1)
Yegappan Lakshmanan34fcb692021-05-25 20:14:00 +02003631
3632 lines =<< trim END
3633 var x = _
3634 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003635 v9.CheckDefAndScriptFailure(lines, 'E1181:', 1)
Bram Moolenaar962c43b2021-04-10 17:18:09 +02003636enddef
3637
Bram Moolenaarbb8a7ce2021-04-10 20:10:26 +02003638def Test_too_many_arguments()
3639 var lines =<< trim END
3640 echo [0, 1, 2]->map(() => 123)
3641 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003642 v9.CheckDefExecAndScriptFailure(lines, 'E1106: 2 arguments too many', 1)
Bram Moolenaarbb8a7ce2021-04-10 20:10:26 +02003643
3644 lines =<< trim END
3645 echo [0, 1, 2]->map((_) => 123)
3646 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003647 v9.CheckDefExecAndScriptFailure(lines, 'E1106: One argument too many', 1)
Bram Moolenaarbb8a7ce2021-04-10 20:10:26 +02003648enddef
Bram Moolenaar077a4232020-12-22 18:33:27 +01003649
Bram Moolenaara6aa1642021-04-23 19:32:23 +02003650def Test_closing_brace_at_start_of_line()
3651 var lines =<< trim END
3652 def Func()
3653 enddef
3654 Func(
3655 )
3656 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003657 v9.CheckDefAndScriptSuccess(lines)
Bram Moolenaara6aa1642021-04-23 19:32:23 +02003658enddef
3659
Bram Moolenaar62aec932022-01-29 21:45:34 +00003660func s:CreateMydict()
Bram Moolenaarb033ee22021-08-15 16:08:36 +02003661 let g:mydict = {}
3662 func g:mydict.afunc()
3663 let g:result = self.key
3664 endfunc
3665endfunc
3666
3667def Test_numbered_function_reference()
3668 CreateMydict()
3669 var output = execute('legacy func g:mydict.afunc')
3670 var funcName = 'g:' .. substitute(output, '.*function \(\d\+\).*', '\1', '')
3671 execute 'function(' .. funcName .. ', [], {key: 42})()'
3672 # check that the function still exists
3673 assert_equal(output, execute('legacy func g:mydict.afunc'))
3674 unlet g:mydict
3675enddef
3676
Bram Moolenaard3a11782022-01-05 16:50:40 +00003677def Test_go_beyond_end_of_cmd()
3678 # this was reading the byte after the end of the line
3679 var lines =<< trim END
3680 def F()
3681 cal
3682 enddef
3683 defcompile
3684 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003685 v9.CheckScriptFailure(lines, 'E476:')
Bram Moolenaard3a11782022-01-05 16:50:40 +00003686enddef
3687
Bram Moolenaar20677332021-06-06 17:02:53 +02003688if has('python3')
3689 def Test_python3_heredoc()
3690 py3 << trim EOF
3691 import vim
3692 vim.vars['didit'] = 'yes'
3693 EOF
3694 assert_equal('yes', g:didit)
3695
3696 python3 << trim EOF
3697 import vim
3698 vim.vars['didit'] = 'again'
3699 EOF
3700 assert_equal('again', g:didit)
3701 enddef
3702endif
3703
3704" This messes up syntax highlight, keep near the end.
3705if has('lua')
3706 def Test_lua_heredoc()
3707 g:d = {}
3708 lua << trim EOF
3709 x = vim.eval('g:d')
3710 x['key'] = 'val'
3711 EOF
3712 assert_equal('val', g:d.key)
3713 enddef
3714endif
3715
Bram Moolenaarf7779c62020-05-03 15:38:16 +02003716
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02003717" vim: ts=8 sw=2 sts=2 expandtab tw=80 fdm=marker