blob: d67dd55b6ee2a9e23c7fa6364675e2ccf4b88f88 [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 Moolenaarad6d9cc2022-08-08 21:43:11 +010077def Test_comment_error()
78 v9.CheckDefFailure(['#{ comment'], 'E1170:')
79enddef
80
Bram Moolenaarb55d6182021-06-08 22:01:53 +020081def Test_compile_error_in_called_function()
82 var lines =<< trim END
83 vim9script
84 var n: number
85 def Foo()
86 &hls = n
87 enddef
88 def Bar()
89 Foo()
90 enddef
91 silent! Foo()
92 Bar()
93 END
Bram Moolenaar62aec932022-01-29 21:45:34 +000094 v9.CheckScriptFailureList(lines, ['E1012:', 'E1191:'])
Bram Moolenaarb55d6182021-06-08 22:01:53 +020095enddef
96
Bram Moolenaar22f17a22021-06-21 20:48:58 +020097def Test_wrong_function_name()
98 var lines =<< trim END
99 vim9script
100 func _Foo()
101 echo 'foo'
102 endfunc
103 END
Bram Moolenaar3787f262022-02-07 21:54:01 +0000104 v9.CheckScriptFailure(lines, 'E1267:')
Bram Moolenaar22f17a22021-06-21 20:48:58 +0200105
106 lines =<< trim END
107 vim9script
108 def _Foo()
109 echo 'foo'
110 enddef
111 END
Bram Moolenaar3787f262022-02-07 21:54:01 +0000112 v9.CheckScriptFailure(lines, 'E1267:')
Bram Moolenaardea5ab02022-02-23 22:12:02 +0000113
114 lines =<< trim END
115 vim9script
116 var Object = {}
117 function Object.Method()
118 endfunction
119 END
120 v9.CheckScriptFailure(lines, 'E1182:')
121
122 lines =<< trim END
123 vim9script
124 var Object = {}
125 def Object.Method()
126 enddef
127 END
128 v9.CheckScriptFailure(lines, 'E1182:')
129
130 lines =<< trim END
131 vim9script
132 g:Object = {}
133 function g:Object.Method()
134 endfunction
135 END
136 v9.CheckScriptFailure(lines, 'E1182:')
137
138 lines =<< trim END
139 let s:Object = {}
140 def Define()
141 function s:Object.Method()
142 endfunction
143 enddef
144 defcompile
145 END
146 v9.CheckScriptFailure(lines, 'E1182:')
147 delfunc g:Define
148
149 lines =<< trim END
150 let s:Object = {}
151 def Define()
152 def Object.Method()
153 enddef
154 enddef
155 defcompile
156 END
157 v9.CheckScriptFailure(lines, 'E1182:')
158 delfunc g:Define
159
160 lines =<< trim END
161 let g:Object = {}
162 def Define()
163 function g:Object.Method()
164 endfunction
165 enddef
166 defcompile
167 END
168 v9.CheckScriptFailure(lines, 'E1182:')
169 delfunc g:Define
Bram Moolenaar22f17a22021-06-21 20:48:58 +0200170enddef
171
Bram Moolenaarf48b2fa2021-04-12 22:02:36 +0200172def Test_autoload_name_mismatch()
173 var dir = 'Xdir/autoload'
174 mkdir(dir, 'p')
175
176 var lines =<< trim END
177 vim9script
Bram Moolenaard8fe6d32022-01-30 18:40:44 +0000178 export def NoFunction()
Bram Moolenaarf48b2fa2021-04-12 22:02:36 +0200179 # comment
180 g:runtime = 'yes'
181 enddef
182 END
183 writefile(lines, dir .. '/script.vim')
184
185 var save_rtp = &rtp
186 exe 'set rtp=' .. getcwd() .. '/Xdir'
187 lines =<< trim END
188 call script#Function()
189 END
Bram Moolenaard8fe6d32022-01-30 18:40:44 +0000190 v9.CheckScriptFailure(lines, 'E117:', 1)
Bram Moolenaarf48b2fa2021-04-12 22:02:36 +0200191
192 &rtp = save_rtp
193 delete(dir, 'rf')
194enddef
195
Bram Moolenaarf0a40692021-06-11 22:05:47 +0200196def Test_autoload_names()
197 var dir = 'Xdir/autoload'
198 mkdir(dir, 'p')
199
200 var lines =<< trim END
201 func foobar#function()
202 return 'yes'
203 endfunc
204 let foobar#var = 'no'
205 END
206 writefile(lines, dir .. '/foobar.vim')
207
208 var save_rtp = &rtp
209 exe 'set rtp=' .. getcwd() .. '/Xdir'
210
211 lines =<< trim END
212 assert_equal('yes', foobar#function())
213 var Function = foobar#function
214 assert_equal('yes', Function())
215
216 assert_equal('no', foobar#var)
217 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000218 v9.CheckDefAndScriptSuccess(lines)
Bram Moolenaarf0a40692021-06-11 22:05:47 +0200219
220 &rtp = save_rtp
221 delete(dir, 'rf')
222enddef
223
Bram Moolenaar88c89c72021-08-14 14:01:05 +0200224def Test_autoload_error_in_script()
225 var dir = 'Xdir/autoload'
226 mkdir(dir, 'p')
227
228 var lines =<< trim END
229 func scripterror#function()
230 let g:called_function = 'yes'
231 endfunc
232 let 0 = 1
233 END
234 writefile(lines, dir .. '/scripterror.vim')
235
236 var save_rtp = &rtp
237 exe 'set rtp=' .. getcwd() .. '/Xdir'
238
239 g:called_function = 'no'
240 # The error in the autoload script cannot be checked with assert_fails(), use
241 # CheckDefSuccess() instead of CheckDefFailure()
242 try
Bram Moolenaar62aec932022-01-29 21:45:34 +0000243 v9.CheckDefSuccess(['scripterror#function()'])
Bram Moolenaar88c89c72021-08-14 14:01:05 +0200244 catch
245 assert_match('E121: Undefined variable: 0', v:exception)
246 endtry
247 assert_equal('no', g:called_function)
248
249 lines =<< trim END
250 func scriptcaught#function()
251 let g:called_function = 'yes'
252 endfunc
253 try
254 let 0 = 1
255 catch
256 let g:caught = v:exception
257 endtry
258 END
259 writefile(lines, dir .. '/scriptcaught.vim')
260
261 g:called_function = 'no'
Bram Moolenaar62aec932022-01-29 21:45:34 +0000262 v9.CheckDefSuccess(['scriptcaught#function()'])
Bram Moolenaar88c89c72021-08-14 14:01:05 +0200263 assert_match('E121: Undefined variable: 0', g:caught)
264 assert_equal('yes', g:called_function)
265
266 &rtp = save_rtp
267 delete(dir, 'rf')
268enddef
269
Bram Moolenaar62aec932022-01-29 21:45:34 +0000270def s:CallRecursive(n: number): number
Bram Moolenaar0ba48e82020-11-17 18:23:19 +0100271 return CallRecursive(n + 1)
272enddef
273
Bram Moolenaar62aec932022-01-29 21:45:34 +0000274def s:CallMapRecursive(l: list<number>): number
Bram Moolenaar2949cfd2020-12-31 21:28:47 +0100275 return map(l, (_, v) => CallMapRecursive([v]))[0]
Bram Moolenaar0ba48e82020-11-17 18:23:19 +0100276enddef
277
278def Test_funcdepth_error()
279 set maxfuncdepth=10
280
281 var caught = false
282 try
283 CallRecursive(1)
284 catch /E132:/
285 caught = true
286 endtry
287 assert_true(caught)
288
289 caught = false
290 try
291 CallMapRecursive([1])
292 catch /E132:/
293 caught = true
294 endtry
295 assert_true(caught)
296
297 set maxfuncdepth&
298enddef
299
Bram Moolenaar5178b1b2021-01-01 18:43:51 +0100300def Test_endfunc_enddef()
301 var lines =<< trim END
302 def Test()
303 echo 'test'
304 endfunc
305 enddef
306 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000307 v9.CheckScriptFailure(lines, 'E1151:', 3)
Bram Moolenaar5178b1b2021-01-01 18:43:51 +0100308
309 lines =<< trim END
310 def Test()
311 func Nested()
312 echo 'test'
313 enddef
314 enddef
315 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000316 v9.CheckScriptFailure(lines, 'E1152:', 4)
Bram Moolenaar49f1e9e2021-03-22 20:49:02 +0100317
318 lines =<< trim END
319 def Ok()
320 echo 'hello'
321 enddef | echo 'there'
322 def Bad()
323 echo 'hello'
324 enddef there
325 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000326 v9.CheckScriptFailure(lines, 'E1173: Text found after enddef: there', 6)
Bram Moolenaar5178b1b2021-01-01 18:43:51 +0100327enddef
328
Bram Moolenaarb8ba9b92021-01-01 18:54:34 +0100329def Test_missing_endfunc_enddef()
330 var lines =<< trim END
331 vim9script
332 def Test()
333 echo 'test'
334 endef
335 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000336 v9.CheckScriptFailure(lines, 'E1057:', 2)
Bram Moolenaarb8ba9b92021-01-01 18:54:34 +0100337
338 lines =<< trim END
339 vim9script
340 func Some()
341 echo 'test'
342 enfffunc
343 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000344 v9.CheckScriptFailure(lines, 'E126:', 2)
Bram Moolenaarb8ba9b92021-01-01 18:54:34 +0100345enddef
346
Bram Moolenaar4efd9942021-01-24 21:14:20 +0100347def Test_white_space_before_paren()
348 var lines =<< trim END
349 vim9script
350 def Test ()
351 echo 'test'
352 enddef
353 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000354 v9.CheckScriptFailure(lines, 'E1068:', 2)
Bram Moolenaar4efd9942021-01-24 21:14:20 +0100355
356 lines =<< trim END
357 vim9script
358 func Test ()
359 echo 'test'
360 endfunc
361 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000362 v9.CheckScriptFailure(lines, 'E1068:', 2)
Bram Moolenaar4efd9942021-01-24 21:14:20 +0100363
364 lines =<< trim END
365 def Test ()
366 echo 'test'
367 enddef
368 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000369 v9.CheckScriptFailure(lines, 'E1068:', 1)
Bram Moolenaar4efd9942021-01-24 21:14:20 +0100370
371 lines =<< trim END
372 func Test ()
373 echo 'test'
374 endfunc
375 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000376 v9.CheckScriptSuccess(lines)
Bram Moolenaar4efd9942021-01-24 21:14:20 +0100377enddef
378
Bram Moolenaar832ea892021-01-08 21:55:26 +0100379def Test_enddef_dict_key()
380 var d = {
381 enddef: 'x',
382 endfunc: 'y',
383 }
384 assert_equal({enddef: 'x', endfunc: 'y'}, d)
385enddef
386
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200387def ReturnString(): string
388 return 'string'
389enddef
390
391def ReturnNumber(): number
392 return 123
393enddef
394
395let g:notNumber = 'string'
396
397def ReturnGlobal(): number
398 return g:notNumber
399enddef
400
401def Test_return_something()
Bram Moolenaar62aec932022-01-29 21:45:34 +0000402 g:ReturnString()->assert_equal('string')
403 g:ReturnNumber()->assert_equal(123)
Bram Moolenaar848fadd2022-01-30 15:28:30 +0000404 assert_fails('g:ReturnGlobal()', 'E1012: Type mismatch; expected number but got string', '', 1, 'ReturnGlobal')
Bram Moolenaaref7aadb2022-01-18 18:46:07 +0000405
406 var lines =<< trim END
407 vim9script
408
409 def Msg()
410 echomsg 'in Msg()...'
411 enddef
412
413 def Func()
414 return Msg()
415 enddef
416 defcompile
417 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000418 v9.CheckScriptFailure(lines, 'E1096:')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200419enddef
420
Bram Moolenaare32e5162021-01-21 20:21:29 +0100421def Test_check_argument_type()
422 var lines =<< trim END
423 vim9script
424 def Val(a: number, b: number): number
425 return 0
426 enddef
427 def Func()
428 var x: any = true
429 Val(0, x)
430 enddef
431 disass Func
432 Func()
433 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000434 v9.CheckScriptFailure(lines, 'E1013: Argument 2: type mismatch, expected number but got bool', 2)
Bram Moolenaare32e5162021-01-21 20:21:29 +0100435enddef
436
Bram Moolenaarefd88552020-06-18 20:50:10 +0200437def Test_missing_return()
Bram Moolenaar62aec932022-01-29 21:45:34 +0000438 v9.CheckDefFailure(['def Missing(): number',
Bram Moolenaarefd88552020-06-18 20:50:10 +0200439 ' if g:cond',
440 ' echo "no return"',
441 ' else',
442 ' return 0',
443 ' endif'
444 'enddef'], 'E1027:')
Bram Moolenaar62aec932022-01-29 21:45:34 +0000445 v9.CheckDefFailure(['def Missing(): number',
Bram Moolenaarefd88552020-06-18 20:50:10 +0200446 ' if g:cond',
447 ' return 1',
448 ' else',
449 ' echo "no return"',
450 ' endif'
451 'enddef'], 'E1027:')
Bram Moolenaar62aec932022-01-29 21:45:34 +0000452 v9.CheckDefFailure(['def Missing(): number',
Bram Moolenaarefd88552020-06-18 20:50:10 +0200453 ' if g:cond',
454 ' return 1',
455 ' else',
456 ' return 2',
457 ' endif'
458 ' return 3'
459 'enddef'], 'E1095:')
460enddef
461
Bram Moolenaar403dc312020-10-17 19:29:51 +0200462def Test_return_bool()
463 var lines =<< trim END
464 vim9script
465 def MenuFilter(id: number, key: string): bool
466 return popup_filter_menu(id, key)
467 enddef
468 def YesnoFilter(id: number, key: string): bool
469 return popup_filter_yesno(id, key)
470 enddef
471 defcompile
472 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000473 v9.CheckScriptSuccess(lines)
Bram Moolenaar403dc312020-10-17 19:29:51 +0200474enddef
475
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200476let s:nothing = 0
477def ReturnNothing()
478 s:nothing = 1
479 if true
480 return
481 endif
482 s:nothing = 2
483enddef
484
485def Test_return_nothing()
Bram Moolenaar62aec932022-01-29 21:45:34 +0000486 g:ReturnNothing()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +0200487 s:nothing->assert_equal(1)
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200488enddef
489
Bram Moolenaar648ea762021-01-15 19:04:32 +0100490def Test_return_invalid()
491 var lines =<< trim END
492 vim9script
493 def Func(): invalid
494 return xxx
495 enddef
496 defcompile
497 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000498 v9.CheckScriptFailure(lines, 'E1010:', 2)
Bram Moolenaar31842cd2021-02-12 22:10:21 +0100499
500 lines =<< trim END
501 vim9script
502 def Test(Fun: func(number): number): list<number>
503 return map([1, 2, 3], (_, i) => Fun(i))
504 enddef
505 defcompile
506 def Inc(nr: number): nr
507 return nr + 2
508 enddef
509 echo Test(Inc)
510 END
511 # doing this twice was leaking memory
Bram Moolenaar62aec932022-01-29 21:45:34 +0000512 v9.CheckScriptFailure(lines, 'E1010:')
513 v9.CheckScriptFailure(lines, 'E1010:')
Bram Moolenaar648ea762021-01-15 19:04:32 +0100514enddef
515
Bram Moolenaarefc084e2021-09-09 22:30:52 +0200516def Test_return_list_any()
Bram Moolenaar114dbda2022-01-03 12:28:03 +0000517 # This used to fail but now the actual list type is checked, and since it has
518 # an item of type string it can be used as list<string>.
Bram Moolenaarefc084e2021-09-09 22:30:52 +0200519 var lines =<< trim END
520 vim9script
521 def Func(): list<string>
522 var l: list<any>
523 l->add('string')
524 return l
525 enddef
526 echo Func()
527 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000528 v9.CheckScriptSuccess(lines)
Bram Moolenaar114dbda2022-01-03 12:28:03 +0000529
Bram Moolenaarefc084e2021-09-09 22:30:52 +0200530 lines =<< trim END
531 vim9script
532 def Func(): list<string>
533 var l: list<any>
534 l += ['string']
535 return l
536 enddef
537 echo Func()
538 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000539 v9.CheckScriptSuccess(lines)
Bram Moolenaarefc084e2021-09-09 22:30:52 +0200540enddef
541
Bram Moolenaar1a572e92022-03-15 12:28:10 +0000542def Test_return_any_two_types()
543 var lines =<< trim END
544 vim9script
545
546 def G(Fn: func(string): any)
547 g:result = Fn("hello")
548 enddef
549
550 def F(a: number, b: string): any
551 echo b
552 if a > 0
553 return 1
554 else
555 return []
556 endif
557 enddef
558
559 G(function(F, [1]))
560 END
561 v9.CheckScriptSuccess(lines)
562 assert_equal(1, g:result)
563 unlet g:result
564enddef
565
Bram Moolenaar62aec932022-01-29 21:45:34 +0000566func s:Increment()
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200567 let g:counter += 1
568endfunc
569
570def Test_call_ufunc_count()
571 g:counter = 1
572 Increment()
573 Increment()
574 Increment()
Bram Moolenaarf5be8cd2020-07-17 20:36:00 +0200575 # works with and without :call
Bram Moolenaarc0c71e92020-09-11 19:09:48 +0200576 g:counter->assert_equal(4)
577 eval g:counter->assert_equal(4)
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200578 unlet g:counter
579enddef
580
Bram Moolenaar1983f1a2022-02-28 20:55:02 +0000581def Test_call_ufunc_failure()
582 var lines =<< trim END
583 vim9script
584 def Tryit()
585 g:Global(1, 2, 3)
586 enddef
587
588 func g:Global(a, b, c)
589 echo a:a a:b a:c
590 endfunc
591
592 defcompile
593
594 func! g:Global(a, b)
595 echo a:a a:b
596 endfunc
597 Tryit()
598 END
599 v9.CheckScriptFailure(lines, 'E118: Too many arguments for function: Global')
600 delfunc g:Global
601
602 lines =<< trim END
603 vim9script
604
605 g:Ref = function('len')
606 def Tryit()
607 g:Ref('x')
608 enddef
609
610 defcompile
611
612 g:Ref = function('add')
613 Tryit()
614 END
615 v9.CheckScriptFailure(lines, 'E119: Not enough arguments for function: add')
616 unlet g:Ref
617enddef
618
Bram Moolenaar62aec932022-01-29 21:45:34 +0000619def s:MyVarargs(arg: string, ...rest: list<string>): string
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +0200620 var res = arg
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200621 for s in rest
622 res ..= ',' .. s
623 endfor
624 return res
625enddef
626
627def Test_call_varargs()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +0200628 MyVarargs('one')->assert_equal('one')
629 MyVarargs('one', 'two')->assert_equal('one,two')
630 MyVarargs('one', 'two', 'three')->assert_equal('one,two,three')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200631enddef
632
Bram Moolenaar01dd6c32021-09-05 16:36:23 +0200633def Test_call_white_space()
Bram Moolenaar62aec932022-01-29 21:45:34 +0000634 v9.CheckDefAndScriptFailure(["call Test ('text')"], ['E476:', 'E1068:'])
Bram Moolenaar01dd6c32021-09-05 16:36:23 +0200635enddef
636
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200637def MyDefaultArgs(name = 'string'): string
638 return name
639enddef
640
Bram Moolenaar62aec932022-01-29 21:45:34 +0000641def s:MyDefaultSecond(name: string, second: bool = true): string
Bram Moolenaare30f64b2020-07-15 19:48:20 +0200642 return second ? name : 'none'
643enddef
644
Bram Moolenaar38a3bfa2021-03-29 22:14:55 +0200645
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200646def Test_call_default_args()
Bram Moolenaar62aec932022-01-29 21:45:34 +0000647 g:MyDefaultArgs()->assert_equal('string')
648 g:MyDefaultArgs(v:none)->assert_equal('string')
649 g:MyDefaultArgs('one')->assert_equal('one')
650 assert_fails('g:MyDefaultArgs("one", "two")', 'E118:', '', 4, 'Test_call_default_args')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200651
Bram Moolenaarc0c71e92020-09-11 19:09:48 +0200652 MyDefaultSecond('test')->assert_equal('test')
653 MyDefaultSecond('test', true)->assert_equal('test')
654 MyDefaultSecond('test', false)->assert_equal('none')
Bram Moolenaare30f64b2020-07-15 19:48:20 +0200655
Bram Moolenaar38a3bfa2021-03-29 22:14:55 +0200656 var lines =<< trim END
657 def MyDefaultThird(name: string, aa = 'aa', bb = 'bb'): string
658 return name .. aa .. bb
659 enddef
660
661 MyDefaultThird('->')->assert_equal('->aabb')
662 MyDefaultThird('->', v:none)->assert_equal('->aabb')
663 MyDefaultThird('->', 'xx')->assert_equal('->xxbb')
664 MyDefaultThird('->', v:none, v:none)->assert_equal('->aabb')
665 MyDefaultThird('->', 'xx', v:none)->assert_equal('->xxbb')
666 MyDefaultThird('->', v:none, 'yy')->assert_equal('->aayy')
667 MyDefaultThird('->', 'xx', 'yy')->assert_equal('->xxyy')
Bram Moolenaare28d9b32021-07-03 18:56:53 +0200668
669 def DefArg(mandatory: any, optional = mandatory): string
670 return mandatory .. optional
671 enddef
672 DefArg(1234)->assert_equal('12341234')
673 DefArg("ok")->assert_equal('okok')
Bram Moolenaar38a3bfa2021-03-29 22:14:55 +0200674 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000675 v9.CheckDefAndScriptSuccess(lines)
Bram Moolenaar38a3bfa2021-03-29 22:14:55 +0200676
Bram Moolenaar62aec932022-01-29 21:45:34 +0000677 v9.CheckScriptFailure(['def Func(arg: number = asdf)', 'enddef', 'defcompile'], 'E1001:')
Bram Moolenaar2d870f82020-12-05 13:41:01 +0100678 delfunc g:Func
Bram Moolenaar62aec932022-01-29 21:45:34 +0000679 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 +0100680 delfunc g:Func
Bram Moolenaar62aec932022-01-29 21:45:34 +0000681 v9.CheckDefFailure(['def Func(x: number = )', 'enddef'], 'E15:')
Bram Moolenaar12bce952021-03-11 20:04:04 +0100682
Bram Moolenaar38a3bfa2021-03-29 22:14:55 +0200683 lines =<< trim END
Bram Moolenaar12bce952021-03-11 20:04:04 +0100684 vim9script
685 def Func(a = b == 0 ? 1 : 2, b = 0)
686 enddef
687 defcompile
688 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000689 v9.CheckScriptFailure(lines, 'E1001: Variable not found: b')
Bram Moolenaar59618fe2021-12-21 12:32:17 +0000690
Bram Moolenaarfa46ead2021-12-22 13:18:39 +0000691 # using script variable requires matching type or type cast when executed
Bram Moolenaar59618fe2021-12-21 12:32:17 +0000692 lines =<< trim END
693 vim9script
694 var a: any
695 def Func(arg: string = a)
696 echo arg
697 enddef
698 defcompile
699 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000700 v9.CheckScriptSuccess(lines + ['a = "text"', 'Func()'])
701 v9.CheckScriptFailure(lines + ['a = 123', 'Func()'], 'E1013: Argument 1: type mismatch, expected string but got number')
Bram Moolenaar59618fe2021-12-21 12:32:17 +0000702
703 # using global variable does not require type cast
704 lines =<< trim END
705 vim9script
706 def Func(arg: string = g:str)
707 echo arg
708 enddef
709 g:str = 'works'
710 Func()
711 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000712 v9.CheckScriptSuccess(lines)
Bram Moolenaar04b12692020-05-04 23:24:44 +0200713enddef
714
Bram Moolenaar62aec932022-01-29 21:45:34 +0000715def s:FuncWithComment( # comment
Bram Moolenaarcef12702021-01-04 14:09:43 +0100716 a: number, #comment
717 b: bool, # comment
718 c: string) #comment
719 assert_equal(4, a)
720 assert_equal(true, b)
721 assert_equal('yes', c)
722enddef
723
724def Test_func_with_comments()
725 FuncWithComment(4, true, 'yes')
726
727 var lines =<< trim END
728 def Func(# comment
729 arg: string)
730 enddef
731 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000732 v9.CheckScriptFailure(lines, 'E125:', 1)
Bram Moolenaarcef12702021-01-04 14:09:43 +0100733
734 lines =<< trim END
735 def Func(
736 arg: string# comment
737 )
738 enddef
739 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000740 v9.CheckScriptFailure(lines, 'E475:', 2)
Bram Moolenaarcef12702021-01-04 14:09:43 +0100741
742 lines =<< trim END
743 def Func(
744 arg: string
745 )# comment
746 enddef
747 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000748 v9.CheckScriptFailure(lines, 'E488:', 3)
Bram Moolenaarcef12702021-01-04 14:09:43 +0100749enddef
750
Bram Moolenaar04b12692020-05-04 23:24:44 +0200751def Test_nested_function()
Bram Moolenaar38453522021-11-28 22:00:12 +0000752 def NestedDef(arg: string): string
Bram Moolenaar04b12692020-05-04 23:24:44 +0200753 return 'nested ' .. arg
754 enddef
Bram Moolenaar38453522021-11-28 22:00:12 +0000755 NestedDef(':def')->assert_equal('nested :def')
756
757 func NestedFunc(arg)
758 return 'nested ' .. a:arg
759 endfunc
760 NestedFunc(':func')->assert_equal('nested :func')
Bram Moolenaar04b12692020-05-04 23:24:44 +0200761
Bram Moolenaar62aec932022-01-29 21:45:34 +0000762 v9.CheckDefFailure(['def Nested()', 'enddef', 'Nested(66)'], 'E118:')
763 v9.CheckDefFailure(['def Nested(arg: string)', 'enddef', 'Nested()'], 'E119:')
Bram Moolenaar0e65d3d2020-05-05 17:53:16 +0200764
Bram Moolenaar62aec932022-01-29 21:45:34 +0000765 v9.CheckDefFailure(['def s:Nested()', 'enddef'], 'E1075:')
766 v9.CheckDefFailure(['def b:Nested()', 'enddef'], 'E1075:')
Bram Moolenaar8b848ca2020-09-10 22:28:01 +0200767
Bram Moolenaar54021752020-12-06 18:50:36 +0100768 var lines =<< trim END
769 def Outer()
770 def Inner()
771 # comment
772 enddef
773 def Inner()
774 enddef
775 enddef
776 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000777 v9.CheckDefFailure(lines, 'E1073:')
Bram Moolenaar54021752020-12-06 18:50:36 +0100778
779 lines =<< trim END
780 def Outer()
781 def Inner()
782 # comment
783 enddef
784 def! Inner()
785 enddef
786 enddef
787 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000788 v9.CheckDefFailure(lines, 'E1117:')
Bram Moolenaar54021752020-12-06 18:50:36 +0100789
Bram Moolenaardb8e5c22021-12-25 19:58:22 +0000790 lines =<< trim END
791 vim9script
792 def Outer()
793 def Inner()
794 g:result = 'ok'
795 enddef
796 Inner()
797 enddef
798 Outer()
799 Inner()
800 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000801 v9.CheckScriptFailure(lines, 'E117: Unknown function: Inner')
Bram Moolenaardb8e5c22021-12-25 19:58:22 +0000802 assert_equal('ok', g:result)
803 unlet g:result
804
Bram Moolenaarf681cfb2022-02-07 20:30:57 +0000805 lines =<< trim END
806 vim9script
807 def Outer()
808 def _Inner()
809 echo 'bad'
810 enddef
Bram Moolenaar3787f262022-02-07 21:54:01 +0000811 _Inner()
Bram Moolenaarf681cfb2022-02-07 20:30:57 +0000812 enddef
813 defcompile
814 END
Bram Moolenaar3787f262022-02-07 21:54:01 +0000815 v9.CheckScriptFailure(lines, 'E1267:')
Bram Moolenaarf681cfb2022-02-07 20:30:57 +0000816
817 lines =<< trim END
818 vim9script
819 def Outer()
820 def g:inner()
821 echo 'bad'
822 enddef
Bram Moolenaar3787f262022-02-07 21:54:01 +0000823 g:inner()
Bram Moolenaarf681cfb2022-02-07 20:30:57 +0000824 enddef
825 defcompile
826 END
Bram Moolenaar3787f262022-02-07 21:54:01 +0000827 v9.CheckScriptFailure(lines, 'E1267:')
828
829 lines =<< trim END
830 vim9script
831 def g:_Func()
832 echo 'bad'
833 enddef
834 END
835 v9.CheckScriptFailure(lines, 'E1267:')
836
837 lines =<< trim END
838 vim9script
Bram Moolenaara749a422022-02-12 19:52:25 +0000839 def _Func()
Bram Moolenaar3787f262022-02-07 21:54:01 +0000840 echo 'bad'
841 enddef
842 END
843 v9.CheckScriptFailure(lines, 'E1267:')
Bram Moolenaarf681cfb2022-02-07 20:30:57 +0000844
Bram Moolenaar54021752020-12-06 18:50:36 +0100845 # nested function inside conditional
Bram Moolenaar54021752020-12-06 18:50:36 +0100846 lines =<< trim END
847 vim9script
848 var thecount = 0
849 if true
850 def Test(): number
851 def TheFunc(): number
852 thecount += 1
853 return thecount
854 enddef
855 return TheFunc()
856 enddef
857 endif
858 defcompile
859 assert_equal(1, Test())
860 assert_equal(2, Test())
861 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000862 v9.CheckScriptSuccess(lines)
Bram Moolenaar8863bda2021-03-17 18:42:08 +0100863
864 # also works when "thecount" is inside the "if" block
865 lines =<< trim END
866 vim9script
867 if true
868 var thecount = 0
869 def Test(): number
870 def TheFunc(): number
871 thecount += 1
872 return thecount
873 enddef
874 return TheFunc()
875 enddef
876 endif
877 defcompile
878 assert_equal(1, Test())
879 assert_equal(2, Test())
880 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000881 v9.CheckScriptSuccess(lines)
Bram Moolenaar4bba16d2021-08-15 19:28:05 +0200882
Bram Moolenaara915fa02022-03-23 11:29:15 +0000883 # nested function with recursive call
884 lines =<< trim END
885 vim9script
886
887 def MyFunc(): number
888 def Fib(n: number): number
889 if n < 2
890 return 1
891 endif
892 return Fib(n - 2) + Fib(n - 1)
893 enddef
894
895 return Fib(5)
896 enddef
897
898 assert_equal(8, MyFunc())
899 END
900 v9.CheckScriptSuccess(lines)
901
Bram Moolenaar4bba16d2021-08-15 19:28:05 +0200902 lines =<< trim END
903 vim9script
904 def Outer()
905 def Inner()
906 echo 'hello'
907 enddef burp
908 enddef
909 defcompile
910 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000911 v9.CheckScriptFailure(lines, 'E1173: Text found after enddef: burp', 3)
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200912enddef
913
Bram Moolenaaradc8e442020-12-31 18:28:18 +0100914def Test_not_nested_function()
915 echo printf('%d',
916 function('len')('xxx'))
917enddef
918
Bram Moolenaaraf8edbb2020-08-01 00:03:09 +0200919func Test_call_default_args_from_func()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +0200920 call MyDefaultArgs()->assert_equal('string')
921 call MyDefaultArgs('one')->assert_equal('one')
Bram Moolenaar9bd5d872020-09-06 21:47:48 +0200922 call assert_fails('call MyDefaultArgs("one", "two")', 'E118:', '', 3, 'Test_call_default_args_from_func')
Bram Moolenaaraf8edbb2020-08-01 00:03:09 +0200923endfunc
924
Bram Moolenaar38ddf332020-07-31 22:05:04 +0200925def Test_nested_global_function()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +0200926 var lines =<< trim END
Bram Moolenaar38ddf332020-07-31 22:05:04 +0200927 vim9script
928 def Outer()
929 def g:Inner(): string
930 return 'inner'
931 enddef
932 enddef
Bram Moolenaaraf8edbb2020-08-01 00:03:09 +0200933 defcompile
934 Outer()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +0200935 g:Inner()->assert_equal('inner')
Bram Moolenaaraf8edbb2020-08-01 00:03:09 +0200936 delfunc g:Inner
937 Outer()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +0200938 g:Inner()->assert_equal('inner')
Bram Moolenaaraf8edbb2020-08-01 00:03:09 +0200939 delfunc g:Inner
940 Outer()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +0200941 g:Inner()->assert_equal('inner')
Bram Moolenaaraf8edbb2020-08-01 00:03:09 +0200942 delfunc g:Inner
Bram Moolenaar38ddf332020-07-31 22:05:04 +0200943 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000944 v9.CheckScriptSuccess(lines)
Bram Moolenaar2c79e9d2020-08-01 18:57:52 +0200945
946 lines =<< trim END
947 vim9script
948 def Outer()
Bram Moolenaar38453522021-11-28 22:00:12 +0000949 func g:Inner()
950 return 'inner'
951 endfunc
952 enddef
953 defcompile
954 Outer()
955 g:Inner()->assert_equal('inner')
956 delfunc g:Inner
957 Outer()
958 g:Inner()->assert_equal('inner')
959 delfunc g:Inner
960 Outer()
961 g:Inner()->assert_equal('inner')
962 delfunc g:Inner
963 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000964 v9.CheckScriptSuccess(lines)
Bram Moolenaar38453522021-11-28 22:00:12 +0000965
966 lines =<< trim END
967 vim9script
968 def Outer()
Bram Moolenaar2c79e9d2020-08-01 18:57:52 +0200969 def g:Inner(): string
970 return 'inner'
971 enddef
972 enddef
973 defcompile
974 Outer()
975 Outer()
976 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000977 v9.CheckScriptFailure(lines, "E122:")
Bram Moolenaarcd45ed02020-12-22 17:35:54 +0100978 delfunc g:Inner
Bram Moolenaarad486a02020-08-01 23:22:18 +0200979
980 lines =<< trim END
981 vim9script
Bram Moolenaar58a52f22020-12-22 18:56:55 +0100982 def Outer()
983 def g:Inner()
Bram Moolenaar2949cfd2020-12-31 21:28:47 +0100984 echo map([1, 2, 3], (_, v) => v + 1)
Bram Moolenaar58a52f22020-12-22 18:56:55 +0100985 enddef
986 g:Inner()
987 enddef
988 Outer()
989 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000990 v9.CheckScriptSuccess(lines)
Bram Moolenaar58a52f22020-12-22 18:56:55 +0100991 delfunc g:Inner
992
993 lines =<< trim END
994 vim9script
Bram Moolenaarad486a02020-08-01 23:22:18 +0200995 def Func()
996 echo 'script'
997 enddef
998 def Outer()
999 def Func()
1000 echo 'inner'
1001 enddef
1002 enddef
1003 defcompile
1004 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001005 v9.CheckScriptFailure(lines, "E1073:", 1)
Bram Moolenaard604d782021-11-20 21:46:20 +00001006
1007 lines =<< trim END
1008 vim9script
1009 def Func()
1010 echo 'script'
1011 enddef
1012 def Func()
1013 echo 'script'
1014 enddef
1015 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001016 v9.CheckScriptFailure(lines, "E1073:", 5)
Bram Moolenaar38ddf332020-07-31 22:05:04 +02001017enddef
1018
Bram Moolenaar6abdcf82020-11-22 18:15:44 +01001019def DefListAll()
1020 def
1021enddef
1022
1023def DefListOne()
1024 def DefListOne
1025enddef
1026
1027def DefListMatches()
1028 def /DefList
1029enddef
1030
1031def Test_nested_def_list()
1032 var funcs = split(execute('call DefListAll()'), "\n")
1033 assert_true(len(funcs) > 10)
1034 assert_true(funcs->index('def DefListAll()') >= 0)
1035
1036 funcs = split(execute('call DefListOne()'), "\n")
1037 assert_equal([' def DefListOne()', '1 def DefListOne', ' enddef'], funcs)
1038
1039 funcs = split(execute('call DefListMatches()'), "\n")
1040 assert_true(len(funcs) >= 3)
1041 assert_true(funcs->index('def DefListAll()') >= 0)
1042 assert_true(funcs->index('def DefListOne()') >= 0)
1043 assert_true(funcs->index('def DefListMatches()') >= 0)
Bram Moolenaar54021752020-12-06 18:50:36 +01001044
1045 var lines =<< trim END
1046 vim9script
1047 def Func()
1048 def +Func+
1049 enddef
1050 defcompile
1051 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001052 v9.CheckScriptFailure(lines, 'E476:', 1)
Bram Moolenaar6abdcf82020-11-22 18:15:44 +01001053enddef
1054
Bram Moolenaare08be092022-02-17 13:08:26 +00001055def Test_global_function_not_found()
1056 var lines =<< trim END
1057 g:Ref = 123
1058 call g:Ref()
1059 END
1060 v9.CheckDefExecAndScriptFailure(lines, ['E117:', 'E1085:'], 2)
1061enddef
1062
Bram Moolenaar333894b2020-08-01 18:53:07 +02001063def Test_global_local_function()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001064 var lines =<< trim END
Bram Moolenaar333894b2020-08-01 18:53:07 +02001065 vim9script
1066 def g:Func(): string
1067 return 'global'
1068 enddef
1069 def Func(): string
1070 return 'local'
1071 enddef
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001072 g:Func()->assert_equal('global')
1073 Func()->assert_equal('local')
Bram Moolenaar2d870f82020-12-05 13:41:01 +01001074 delfunc g:Func
Bram Moolenaar333894b2020-08-01 18:53:07 +02001075 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001076 v9.CheckScriptSuccess(lines)
Bram Moolenaar035d6e92020-08-11 22:30:42 +02001077
1078 lines =<< trim END
1079 vim9script
1080 def g:Funcy()
1081 echo 'funcy'
1082 enddef
Bram Moolenaara749a422022-02-12 19:52:25 +00001083 Funcy()
Bram Moolenaar035d6e92020-08-11 22:30:42 +02001084 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001085 v9.CheckScriptFailure(lines, 'E117:')
Bram Moolenaar333894b2020-08-01 18:53:07 +02001086enddef
1087
Bram Moolenaar0f769812020-09-12 18:32:34 +02001088def Test_local_function_shadows_global()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001089 var lines =<< trim END
Bram Moolenaar0f769812020-09-12 18:32:34 +02001090 vim9script
1091 def g:Gfunc(): string
1092 return 'global'
1093 enddef
1094 def AnotherFunc(): number
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001095 var Gfunc = function('len')
Bram Moolenaar0f769812020-09-12 18:32:34 +02001096 return Gfunc('testing')
1097 enddef
1098 g:Gfunc()->assert_equal('global')
1099 AnotherFunc()->assert_equal(7)
1100 delfunc g:Gfunc
1101 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001102 v9.CheckScriptSuccess(lines)
Bram Moolenaar0f769812020-09-12 18:32:34 +02001103
1104 lines =<< trim END
1105 vim9script
1106 def g:Func(): string
1107 return 'global'
1108 enddef
1109 def AnotherFunc()
1110 g:Func = function('len')
1111 enddef
1112 AnotherFunc()
1113 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001114 v9.CheckScriptFailure(lines, 'E705:')
Bram Moolenaar0f769812020-09-12 18:32:34 +02001115 delfunc g:Func
Bram Moolenaar0865b152021-04-05 15:38:51 +02001116
Bram Moolenaar62aec932022-01-29 21:45:34 +00001117 # global function is not found with g: prefix
Bram Moolenaar0865b152021-04-05 15:38:51 +02001118 lines =<< trim END
1119 vim9script
1120 def g:Func(): string
1121 return 'global'
1122 enddef
1123 def AnotherFunc(): string
1124 return Func()
1125 enddef
1126 assert_equal('global', AnotherFunc())
Bram Moolenaar0865b152021-04-05 15:38:51 +02001127 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001128 v9.CheckScriptFailure(lines, 'E117:')
1129 delfunc g:Func
Bram Moolenaar0865b152021-04-05 15:38:51 +02001130
1131 lines =<< trim END
1132 vim9script
1133 def g:Func(): string
1134 return 'global'
1135 enddef
Bram Moolenaar848fadd2022-01-30 15:28:30 +00001136 assert_equal('global', g:Func())
Bram Moolenaar0865b152021-04-05 15:38:51 +02001137 delfunc g:Func
1138 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001139 v9.CheckScriptSuccess(lines)
Bram Moolenaar58493cf2022-01-06 12:23:30 +00001140
1141 # This does not shadow "i" which is visible only inside the for loop
1142 lines =<< trim END
1143 vim9script
1144
1145 def Foo(i: number)
1146 echo i
1147 enddef
1148
1149 for i in range(3)
1150 # Foo() is compiled here
1151 Foo(i)
1152 endfor
1153 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001154 v9.CheckScriptSuccess(lines)
Bram Moolenaar0f769812020-09-12 18:32:34 +02001155enddef
1156
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001157func TakesOneArg(arg)
1158 echo a:arg
1159endfunc
1160
1161def Test_call_wrong_args()
Bram Moolenaar62aec932022-01-29 21:45:34 +00001162 v9.CheckDefFailure(['g:TakesOneArg()'], 'E119:')
1163 v9.CheckDefFailure(['g:TakesOneArg(11, 22)'], 'E118:')
1164 v9.CheckDefFailure(['bufnr(xxx)'], 'E1001:')
1165 v9.CheckScriptFailure(['def Func(Ref: func(s: string))'], 'E475:')
Bram Moolenaaree8580e2020-08-28 17:19:07 +02001166
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001167 var lines =<< trim END
Bram Moolenaaree8580e2020-08-28 17:19:07 +02001168 vim9script
1169 def Func(s: string)
1170 echo s
1171 enddef
1172 Func([])
1173 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001174 v9.CheckScriptFailure(lines, 'E1013: Argument 1: type mismatch, expected string but got list<unknown>', 5)
Bram Moolenaarb185a402020-09-18 22:42:00 +02001175
Bram Moolenaar9a015112021-12-31 14:06:45 +00001176 # argument name declared earlier is found when declaring a function
Bram Moolenaarb185a402020-09-18 22:42:00 +02001177 lines =<< trim END
1178 vim9script
Bram Moolenaarb4893b82021-02-21 22:20:24 +01001179 var name = 'piet'
1180 def FuncOne(name: string)
Bram Moolenaar3a5988c2022-02-08 19:23:35 +00001181 echo name
Bram Moolenaarb4893b82021-02-21 22:20:24 +01001182 enddef
1183 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001184 v9.CheckScriptFailure(lines, 'E1168:')
Bram Moolenaarb4893b82021-02-21 22:20:24 +01001185
Bram Moolenaar3a5988c2022-02-08 19:23:35 +00001186 # same, inside the same block
1187 lines =<< trim END
1188 vim9script
1189 if true
1190 var name = 'piet'
1191 def FuncOne(name: string)
1192 echo name
1193 enddef
1194 endif
1195 END
1196 v9.CheckScriptFailure(lines, 'E1168:')
1197
1198 # variable in other block is OK
1199 lines =<< trim END
1200 vim9script
1201 if true
1202 var name = 'piet'
1203 endif
1204 def FuncOne(name: string)
1205 echo name
1206 enddef
1207 END
1208 v9.CheckScriptSuccess(lines)
1209
Bram Moolenaardce24412022-02-08 20:35:30 +00001210 # with another variable in another block
1211 lines =<< trim END
1212 vim9script
1213 if true
1214 var name = 'piet'
1215 # define a function so that the variable isn't cleared
1216 def GetItem(): string
1217 return item
1218 enddef
1219 endif
1220 if true
1221 var name = 'peter'
1222 def FuncOne(name: string)
1223 echo name
1224 enddef
1225 endif
1226 END
1227 v9.CheckScriptFailure(lines, 'E1168:')
1228
1229 # only variable in another block is OK
1230 lines =<< trim END
1231 vim9script
1232 if true
1233 var name = 'piet'
1234 # define a function so that the variable isn't cleared
1235 def GetItem(): string
1236 return item
1237 enddef
1238 endif
1239 if true
1240 def FuncOne(name: string)
1241 echo name
1242 enddef
1243 endif
1244 END
1245 v9.CheckScriptSuccess(lines)
1246
Bram Moolenaar9a015112021-12-31 14:06:45 +00001247 # argument name declared later is only found when compiling
1248 lines =<< trim END
1249 vim9script
1250 def FuncOne(name: string)
1251 echo nr
1252 enddef
1253 var name = 'piet'
1254 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001255 v9.CheckScriptSuccess(lines)
1256 v9.CheckScriptFailure(lines + ['defcompile'], 'E1168:')
Bram Moolenaar9a015112021-12-31 14:06:45 +00001257
Bram Moolenaarb4893b82021-02-21 22:20:24 +01001258 lines =<< trim END
1259 vim9script
Bram Moolenaarb185a402020-09-18 22:42:00 +02001260 def FuncOne(nr: number)
1261 echo nr
1262 enddef
1263 def FuncTwo()
1264 FuncOne()
1265 enddef
1266 defcompile
1267 END
1268 writefile(lines, 'Xscript')
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001269 var didCatch = false
Bram Moolenaarb185a402020-09-18 22:42:00 +02001270 try
1271 source Xscript
1272 catch
1273 assert_match('E119: Not enough arguments for function: <SNR>\d\+_FuncOne', v:exception)
1274 assert_match('Xscript\[8\]..function <SNR>\d\+_FuncTwo, line 1', v:throwpoint)
1275 didCatch = true
1276 endtry
1277 assert_true(didCatch)
1278
1279 lines =<< trim END
1280 vim9script
1281 def FuncOne(nr: number)
1282 echo nr
1283 enddef
1284 def FuncTwo()
1285 FuncOne(1, 2)
1286 enddef
1287 defcompile
1288 END
1289 writefile(lines, 'Xscript')
1290 didCatch = false
1291 try
1292 source Xscript
1293 catch
1294 assert_match('E118: Too many arguments for function: <SNR>\d\+_FuncOne', v:exception)
1295 assert_match('Xscript\[8\]..function <SNR>\d\+_FuncTwo, line 1', v:throwpoint)
1296 didCatch = true
1297 endtry
1298 assert_true(didCatch)
1299
1300 delete('Xscript')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001301enddef
1302
Bram Moolenaar50824712020-12-20 21:10:17 +01001303def Test_call_funcref_wrong_args()
1304 var head =<< trim END
1305 vim9script
1306 def Func3(a1: string, a2: number, a3: list<number>)
1307 echo a1 .. a2 .. a3[0]
1308 enddef
1309 def Testme()
1310 var funcMap: dict<func> = {func: Func3}
1311 END
1312 var tail =<< trim END
1313 enddef
1314 Testme()
1315 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001316 v9.CheckScriptSuccess(head + ["funcMap['func']('str', 123, [1, 2, 3])"] + tail)
Bram Moolenaar50824712020-12-20 21:10:17 +01001317
Bram Moolenaar62aec932022-01-29 21:45:34 +00001318 v9.CheckScriptFailure(head + ["funcMap['func']('str', 123)"] + tail, 'E119:')
1319 v9.CheckScriptFailure(head + ["funcMap['func']('str', 123, [1], 4)"] + tail, 'E118:')
Bram Moolenaar32b3f822021-01-06 21:59:39 +01001320
1321 var lines =<< trim END
1322 vim9script
1323 var Ref: func(number): any
1324 Ref = (j) => !j
1325 echo Ref(false)
1326 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001327 v9.CheckScriptFailure(lines, 'E1013: Argument 1: type mismatch, expected number but got bool', 4)
Bram Moolenaar32b3f822021-01-06 21:59:39 +01001328
1329 lines =<< trim END
1330 vim9script
1331 var Ref: func(number): any
1332 Ref = (j) => !j
1333 call Ref(false)
1334 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001335 v9.CheckScriptFailure(lines, 'E1013: Argument 1: type mismatch, expected number but got bool', 4)
Bram Moolenaar50824712020-12-20 21:10:17 +01001336enddef
1337
Bram Moolenaarb4d16cb2020-11-05 18:45:46 +01001338def Test_call_lambda_args()
Bram Moolenaar2a389082021-04-09 20:24:31 +02001339 var lines =<< trim END
1340 var Callback = (..._) => 'anything'
1341 assert_equal('anything', Callback())
1342 assert_equal('anything', Callback(1))
1343 assert_equal('anything', Callback('a', 2))
Bram Moolenaar1088b692021-04-09 22:12:44 +02001344
1345 assert_equal('xyz', ((a: string): string => a)('xyz'))
Bram Moolenaar2a389082021-04-09 20:24:31 +02001346 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001347 v9.CheckDefAndScriptSuccess(lines)
Bram Moolenaar2a389082021-04-09 20:24:31 +02001348
Bram Moolenaar62aec932022-01-29 21:45:34 +00001349 v9.CheckDefFailure(['echo ((i) => 0)()'],
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01001350 'E119: Not enough arguments for function: ((i) => 0)()')
Bram Moolenaarb4d16cb2020-11-05 18:45:46 +01001351
Bram Moolenaar2a389082021-04-09 20:24:31 +02001352 lines =<< trim END
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01001353 var Ref = (x: number, y: number) => x + y
Bram Moolenaarb4d16cb2020-11-05 18:45:46 +01001354 echo Ref(1, 'x')
1355 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001356 v9.CheckDefFailure(lines, 'E1013: Argument 2: type mismatch, expected number but got string')
Bram Moolenaare68b02a2021-01-03 13:09:51 +01001357
1358 lines =<< trim END
1359 var Ref: func(job, string, number)
1360 Ref = (x, y) => 0
1361 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001362 v9.CheckDefAndScriptFailure(lines, 'E1012:')
Bram Moolenaare68b02a2021-01-03 13:09:51 +01001363
1364 lines =<< trim END
1365 var Ref: func(job, string)
1366 Ref = (x, y, z) => 0
1367 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001368 v9.CheckDefAndScriptFailure(lines, 'E1012:')
Bram Moolenaar057e84a2021-02-28 16:55:11 +01001369
1370 lines =<< trim END
1371 var one = 1
1372 var l = [1, 2, 3]
1373 echo map(l, (one) => one)
1374 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001375 v9.CheckDefFailure(lines, 'E1167:')
1376 v9.CheckScriptFailure(['vim9script'] + lines, 'E1168:')
Bram Moolenaar057e84a2021-02-28 16:55:11 +01001377
1378 lines =<< trim END
Bram Moolenaar14ded112021-06-26 19:25:49 +02001379 var Ref: func(any, ?any): bool
1380 Ref = (_, y = 1) => false
1381 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001382 v9.CheckDefAndScriptFailure(lines, 'E1172:')
Bram Moolenaar14ded112021-06-26 19:25:49 +02001383
1384 lines =<< trim END
Bram Moolenaar015cf102021-06-26 21:52:02 +02001385 var a = 0
1386 var b = (a == 0 ? 1 : 2)
1387 assert_equal(1, b)
Bram Moolenaar98f9a5f2021-06-26 22:22:38 +02001388 var txt = 'a'
1389 b = (txt =~ 'x' ? 1 : 2)
1390 assert_equal(2, b)
Bram Moolenaar015cf102021-06-26 21:52:02 +02001391 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001392 v9.CheckDefAndScriptSuccess(lines)
Bram Moolenaar015cf102021-06-26 21:52:02 +02001393
1394 lines =<< trim END
Bram Moolenaar057e84a2021-02-28 16:55:11 +01001395 def ShadowLocal()
1396 var one = 1
1397 var l = [1, 2, 3]
1398 echo map(l, (one) => one)
1399 enddef
1400 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001401 v9.CheckDefFailure(lines, 'E1167:')
Bram Moolenaar057e84a2021-02-28 16:55:11 +01001402
1403 lines =<< trim END
1404 def Shadowarg(one: number)
1405 var l = [1, 2, 3]
1406 echo map(l, (one) => one)
1407 enddef
1408 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001409 v9.CheckDefFailure(lines, 'E1167:')
Bram Moolenaar767034c2021-04-09 17:24:52 +02001410
1411 lines =<< trim END
1412 echo ((a) => a)('aa', 'bb')
1413 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001414 v9.CheckDefAndScriptFailure(lines, 'E118:', 1)
Bram Moolenaarc4c56422021-07-21 20:38:46 +02001415
1416 lines =<< trim END
1417 echo 'aa'->((a) => a)('bb')
1418 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001419 v9.CheckDefFailure(lines, 'E118: Too many arguments for function: ->((a) => a)(''bb'')', 1)
1420 v9.CheckScriptFailure(['vim9script'] + lines, 'E118: Too many arguments for function: <lambda>', 2)
Bram Moolenaarb4d16cb2020-11-05 18:45:46 +01001421enddef
1422
Bram Moolenaara755fdb2021-11-20 21:35:41 +00001423def Test_lambda_line_nr()
1424 var lines =<< trim END
1425 vim9script
1426 # comment
1427 # comment
1428 var id = timer_start(1'000, (_) => 0)
1429 var out = execute('verbose ' .. timer_info(id)[0].callback
1430 ->string()
1431 ->substitute("('\\|')", ' ', 'g'))
1432 assert_match('Last set from .* line 4', out)
1433 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001434 v9.CheckScriptSuccess(lines)
Bram Moolenaara755fdb2021-11-20 21:35:41 +00001435enddef
1436
Bram Moolenaar5f91e742021-03-17 21:29:29 +01001437def FilterWithCond(x: string, Cond: func(string): bool): bool
1438 return Cond(x)
1439enddef
1440
Bram Moolenaar0346b792021-01-31 22:18:29 +01001441def Test_lambda_return_type()
1442 var lines =<< trim END
1443 var Ref = (): => 123
1444 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001445 v9.CheckDefAndScriptFailure(lines, 'E1157:', 1)
Bram Moolenaar5f91e742021-03-17 21:29:29 +01001446
Yegappan Lakshmanan611728f2021-05-24 15:15:47 +02001447 # no space before the return type
1448 lines =<< trim END
1449 var Ref = (x):number => x + 1
1450 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001451 v9.CheckDefAndScriptFailure(lines, 'E1069:', 1)
Yegappan Lakshmanan611728f2021-05-24 15:15:47 +02001452
Bram Moolenaar5f91e742021-03-17 21:29:29 +01001453 # this works
1454 for x in ['foo', 'boo']
Bram Moolenaar62aec932022-01-29 21:45:34 +00001455 echo g:FilterWithCond(x, (v) => v =~ '^b')
Bram Moolenaar5f91e742021-03-17 21:29:29 +01001456 endfor
1457
1458 # this fails
1459 lines =<< trim END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001460 echo g:FilterWithCond('foo', (v) => v .. '^b')
Bram Moolenaar5f91e742021-03-17 21:29:29 +01001461 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001462 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 +02001463
1464 lines =<< trim END
1465 var Lambda1 = (x) => {
1466 return x
1467 }
1468 assert_equal('asdf', Lambda1('asdf'))
1469 var Lambda2 = (x): string => {
1470 return x
1471 }
1472 assert_equal('foo', Lambda2('foo'))
1473 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001474 v9.CheckDefAndScriptSuccess(lines)
Bram Moolenaara9931532021-06-12 15:58:16 +02001475
1476 lines =<< trim END
1477 var Lambda = (x): string => {
1478 return x
1479 }
1480 echo Lambda(['foo'])
1481 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001482 v9.CheckDefExecAndScriptFailure(lines, 'E1012:')
Bram Moolenaar0346b792021-01-31 22:18:29 +01001483enddef
1484
Bram Moolenaar709664c2020-12-12 14:33:41 +01001485def Test_lambda_uses_assigned_var()
Bram Moolenaar62aec932022-01-29 21:45:34 +00001486 v9.CheckDefSuccess([
Bram Moolenaar709664c2020-12-12 14:33:41 +01001487 'var x: any = "aaa"'
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01001488 'x = filter(["bbb"], (_, v) => v =~ x)'])
Bram Moolenaar709664c2020-12-12 14:33:41 +01001489enddef
1490
Bram Moolenaar18062fc2021-03-05 21:35:47 +01001491def Test_pass_legacy_lambda_to_def_func()
1492 var lines =<< trim END
1493 vim9script
1494 func Foo()
1495 eval s:Bar({x -> 0})
1496 endfunc
1497 def Bar(y: any)
1498 enddef
1499 Foo()
1500 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001501 v9.CheckScriptSuccess(lines)
Bram Moolenaar831bdf82021-06-22 19:32:17 +02001502
1503 lines =<< trim END
1504 vim9script
Bram Moolenaar1d9cef72022-03-17 16:30:03 +00001505 def g:TestFunc(F: func)
Bram Moolenaar831bdf82021-06-22 19:32:17 +02001506 enddef
1507 legacy call g:TestFunc({-> 0})
1508 delfunc g:TestFunc
1509
Bram Moolenaar1d9cef72022-03-17 16:30:03 +00001510 def g:TestFunc(F: func(number))
Bram Moolenaar831bdf82021-06-22 19:32:17 +02001511 enddef
1512 legacy call g:TestFunc({nr -> 0})
1513 delfunc g:TestFunc
1514 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001515 v9.CheckScriptSuccess(lines)
Bram Moolenaar18062fc2021-03-05 21:35:47 +01001516enddef
1517
Bram Moolenaar844fb642021-10-23 13:32:30 +01001518def Test_lambda_in_reduce_line_break()
1519 # this was using freed memory
1520 var lines =<< trim END
1521 vim9script
1522 const result: dict<number> =
1523 ['Bob', 'Sam', 'Cat', 'Bob', 'Cat', 'Cat']
1524 ->reduce((acc, val) => {
1525 if has_key(acc, val)
1526 acc[val] += 1
1527 return acc
1528 else
1529 acc[val] = 1
1530 return acc
1531 endif
1532 }, {})
1533 assert_equal({Bob: 2, Sam: 1, Cat: 3}, result)
1534 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001535 v9.CheckScriptSuccess(lines)
Bram Moolenaar844fb642021-10-23 13:32:30 +01001536enddef
1537
Bram Moolenaardcb53be2021-12-09 14:23:43 +00001538def Test_set_opfunc_to_lambda()
1539 var lines =<< trim END
1540 vim9script
1541 nnoremap <expr> <F4> <SID>CountSpaces() .. '_'
1542 def CountSpaces(type = ''): string
1543 if type == ''
1544 &operatorfunc = (t) => CountSpaces(t)
1545 return 'g@'
1546 endif
1547 normal! '[V']y
1548 g:result = getreg('"')->count(' ')
1549 return ''
1550 enddef
1551 new
1552 'a b c d e'->setline(1)
1553 feedkeys("\<F4>", 'x')
1554 assert_equal(4, g:result)
1555 bwipe!
1556 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001557 v9.CheckScriptSuccess(lines)
Bram Moolenaardcb53be2021-12-09 14:23:43 +00001558enddef
1559
Bram Moolenaaref082e12021-12-12 21:02:03 +00001560def Test_set_opfunc_to_global_function()
1561 var lines =<< trim END
1562 vim9script
1563 def g:CountSpaces(type = ''): string
1564 normal! '[V']y
1565 g:result = getreg('"')->count(' ')
1566 return ''
1567 enddef
Bram Moolenaarb15cf442021-12-16 15:49:43 +00001568 # global function works at script level
Bram Moolenaaref082e12021-12-12 21:02:03 +00001569 &operatorfunc = g:CountSpaces
1570 new
1571 'a b c d e'->setline(1)
1572 feedkeys("g@_", 'x')
1573 assert_equal(4, g:result)
Bram Moolenaarb15cf442021-12-16 15:49:43 +00001574
1575 &operatorfunc = ''
1576 g:result = 0
1577 # global function works in :def function
1578 def Func()
1579 &operatorfunc = g:CountSpaces
1580 enddef
1581 Func()
1582 feedkeys("g@_", 'x')
1583 assert_equal(4, g:result)
1584
Bram Moolenaaref082e12021-12-12 21:02:03 +00001585 bwipe!
1586 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001587 v9.CheckScriptSuccess(lines)
Bram Moolenaaref082e12021-12-12 21:02:03 +00001588 &operatorfunc = ''
1589enddef
1590
Bram Moolenaar33b968d2021-12-13 11:31:04 +00001591def Test_use_script_func_name_with_prefix()
1592 var lines =<< trim END
1593 vim9script
Bram Moolenaara749a422022-02-12 19:52:25 +00001594 func g:Getit()
Bram Moolenaar33b968d2021-12-13 11:31:04 +00001595 return 'it'
1596 endfunc
Bram Moolenaara749a422022-02-12 19:52:25 +00001597 var Fn = g:Getit
Bram Moolenaar33b968d2021-12-13 11:31:04 +00001598 assert_equal('it', Fn())
1599 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001600 v9.CheckScriptSuccess(lines)
Bram Moolenaar33b968d2021-12-13 11:31:04 +00001601enddef
1602
Bram Moolenaardd297bc2021-12-10 10:37:38 +00001603def Test_lambda_type_allocated()
1604 # Check that unreferencing a partial using a lambda can use the variable type
1605 # after the lambda has been freed and does not leak memory.
1606 var lines =<< trim END
1607 vim9script
1608
1609 func MyomniFunc1(val, findstart, base)
1610 return a:findstart ? 0 : []
1611 endfunc
1612
1613 var Lambda = (a, b) => MyomniFunc1(19, a, b)
1614 &omnifunc = Lambda
1615 Lambda = (a, b) => MyomniFunc1(20, a, b)
1616 &omnifunc = string(Lambda)
1617 Lambda = (a, b) => strlen(a)
1618 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001619 v9.CheckScriptSuccess(lines)
Bram Moolenaardd297bc2021-12-10 10:37:38 +00001620enddef
1621
Bram Moolenaara7583c42022-05-07 21:14:05 +01001622def Test_define_lambda_in_execute()
1623 var lines =<< trim [CODE]
1624 vim9script
1625
1626 def BuildFuncMultiLine(): func
1627 var x =<< trim END
1628 g:SomeRandomFunc = (d: dict<any>) => {
1629 return d.k1 + d.k2
1630 }
1631 END
1632 execute(x)
1633 return g:SomeRandomFunc
1634 enddef
1635 var ResultPlus = BuildFuncMultiLine()
1636 assert_equal(7, ResultPlus({k1: 3, k2: 4}))
1637 [CODE]
1638 v9.CheckScriptSuccess(lines)
1639 unlet g:SomeRandomFunc
1640enddef
1641
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001642" Default arg and varargs
1643def MyDefVarargs(one: string, two = 'foo', ...rest: list<string>): string
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001644 var res = one .. ',' .. two
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001645 for s in rest
1646 res ..= ',' .. s
1647 endfor
1648 return res
1649enddef
1650
1651def Test_call_def_varargs()
Bram Moolenaar62aec932022-01-29 21:45:34 +00001652 assert_fails('g:MyDefVarargs()', 'E119:', '', 1, 'Test_call_def_varargs')
1653 g:MyDefVarargs('one')->assert_equal('one,foo')
1654 g:MyDefVarargs('one', 'two')->assert_equal('one,two')
1655 g:MyDefVarargs('one', 'two', 'three')->assert_equal('one,two,three')
1656 v9.CheckDefFailure(['g:MyDefVarargs("one", 22)'],
Bram Moolenaar77072282020-09-16 17:55:40 +02001657 'E1013: Argument 2: type mismatch, expected string but got number')
Bram Moolenaar62aec932022-01-29 21:45:34 +00001658 v9.CheckDefFailure(['g:MyDefVarargs("one", "two", 123)'],
Bram Moolenaar77072282020-09-16 17:55:40 +02001659 'E1013: Argument 3: type mismatch, expected string but got number')
Bram Moolenaar24aa48b2020-07-25 16:33:02 +02001660
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001661 var lines =<< trim END
Bram Moolenaar24aa48b2020-07-25 16:33:02 +02001662 vim9script
1663 def Func(...l: list<string>)
1664 echo l
1665 enddef
1666 Func('a', 'b', 'c')
1667 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001668 v9.CheckScriptSuccess(lines)
Bram Moolenaar24aa48b2020-07-25 16:33:02 +02001669
1670 lines =<< trim END
1671 vim9script
1672 def Func(...l: list<string>)
1673 echo l
1674 enddef
1675 Func()
1676 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001677 v9.CheckScriptSuccess(lines)
Bram Moolenaar24aa48b2020-07-25 16:33:02 +02001678
1679 lines =<< trim END
1680 vim9script
Bram Moolenaar2a389082021-04-09 20:24:31 +02001681 def Func(...l: list<any>)
Bram Moolenaar2f8cbc42020-09-16 17:22:59 +02001682 echo l
1683 enddef
1684 Func(0)
1685 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001686 v9.CheckScriptSuccess(lines)
Bram Moolenaar2f8cbc42020-09-16 17:22:59 +02001687
1688 lines =<< trim END
1689 vim9script
Bram Moolenaar2a389082021-04-09 20:24:31 +02001690 def Func(...l: any)
1691 echo l
1692 enddef
1693 Func(0)
1694 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001695 v9.CheckScriptFailure(lines, 'E1180:', 2)
Bram Moolenaar2a389082021-04-09 20:24:31 +02001696
1697 lines =<< trim END
1698 vim9script
Bram Moolenaar28022722020-09-21 22:02:49 +02001699 def Func(..._l: list<string>)
1700 echo _l
1701 enddef
1702 Func('a', 'b', 'c')
1703 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001704 v9.CheckScriptSuccess(lines)
Bram Moolenaar28022722020-09-21 22:02:49 +02001705
1706 lines =<< trim END
1707 vim9script
Bram Moolenaar24aa48b2020-07-25 16:33:02 +02001708 def Func(...l: list<string>)
1709 echo l
1710 enddef
1711 Func(1, 2, 3)
1712 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001713 v9.CheckScriptFailure(lines, 'E1013: Argument 1: type mismatch')
Bram Moolenaar24aa48b2020-07-25 16:33:02 +02001714
1715 lines =<< trim END
1716 vim9script
1717 def Func(...l: list<string>)
1718 echo l
1719 enddef
1720 Func('a', 9)
1721 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001722 v9.CheckScriptFailure(lines, 'E1013: Argument 2: type mismatch')
Bram Moolenaar24aa48b2020-07-25 16:33:02 +02001723
1724 lines =<< trim END
1725 vim9script
1726 def Func(...l: list<string>)
1727 echo l
1728 enddef
1729 Func(1, 'a')
1730 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001731 v9.CheckScriptFailure(lines, 'E1013: Argument 1: type mismatch')
Bram Moolenaar4f53b792021-02-07 15:59:49 +01001732
1733 lines =<< trim END
1734 vim9script
1735 def Func( # some comment
1736 ...l = []
1737 )
1738 echo l
1739 enddef
1740 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001741 v9.CheckScriptFailure(lines, 'E1160:')
Bram Moolenaar6ce46b92021-08-07 15:35:36 +02001742
1743 lines =<< trim END
1744 vim9script
1745 def DoIt()
1746 g:Later('')
1747 enddef
1748 defcompile
1749 def g:Later(...l: list<number>)
1750 enddef
1751 DoIt()
1752 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001753 v9.CheckScriptFailure(lines, 'E1013: Argument 1: type mismatch, expected number but got string')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001754enddef
1755
Bram Moolenaar1378fbc2020-04-11 20:50:33 +02001756let s:value = ''
1757
1758def FuncOneDefArg(opt = 'text')
1759 s:value = opt
1760enddef
1761
1762def FuncTwoDefArg(nr = 123, opt = 'text'): string
1763 return nr .. opt
1764enddef
1765
1766def FuncVarargs(...arg: list<string>): string
1767 return join(arg, ',')
1768enddef
1769
1770def Test_func_type_varargs()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001771 var RefDefArg: func(?string)
Bram Moolenaar848fadd2022-01-30 15:28:30 +00001772 RefDefArg = g:FuncOneDefArg
Bram Moolenaar1378fbc2020-04-11 20:50:33 +02001773 RefDefArg()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001774 s:value->assert_equal('text')
Bram Moolenaar1378fbc2020-04-11 20:50:33 +02001775 RefDefArg('some')
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001776 s:value->assert_equal('some')
Bram Moolenaar1378fbc2020-04-11 20:50:33 +02001777
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001778 var RefDef2Arg: func(?number, ?string): string
Bram Moolenaar848fadd2022-01-30 15:28:30 +00001779 RefDef2Arg = g:FuncTwoDefArg
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001780 RefDef2Arg()->assert_equal('123text')
1781 RefDef2Arg(99)->assert_equal('99text')
1782 RefDef2Arg(77, 'some')->assert_equal('77some')
Bram Moolenaar1378fbc2020-04-11 20:50:33 +02001783
Bram Moolenaar62aec932022-01-29 21:45:34 +00001784 v9.CheckDefFailure(['var RefWrong: func(string?)'], 'E1010:')
1785 v9.CheckDefFailure(['var RefWrong: func(?string, string)'], 'E1007:')
Bram Moolenaar1378fbc2020-04-11 20:50:33 +02001786
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001787 var RefVarargs: func(...list<string>): string
Bram Moolenaar848fadd2022-01-30 15:28:30 +00001788 RefVarargs = g:FuncVarargs
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001789 RefVarargs()->assert_equal('')
1790 RefVarargs('one')->assert_equal('one')
1791 RefVarargs('one', 'two')->assert_equal('one,two')
Bram Moolenaar1378fbc2020-04-11 20:50:33 +02001792
Bram Moolenaar62aec932022-01-29 21:45:34 +00001793 v9.CheckDefFailure(['var RefWrong: func(...list<string>, string)'], 'E110:')
1794 v9.CheckDefFailure(['var RefWrong: func(...list<string>, ?string)'], 'E110:')
Bram Moolenaar1378fbc2020-04-11 20:50:33 +02001795enddef
1796
Bram Moolenaar0b76b422020-04-07 22:05:08 +02001797" Only varargs
1798def MyVarargsOnly(...args: list<string>): string
1799 return join(args, ',')
1800enddef
1801
1802def Test_call_varargs_only()
Bram Moolenaar62aec932022-01-29 21:45:34 +00001803 g:MyVarargsOnly()->assert_equal('')
1804 g:MyVarargsOnly('one')->assert_equal('one')
1805 g:MyVarargsOnly('one', 'two')->assert_equal('one,two')
1806 v9.CheckDefFailure(['g:MyVarargsOnly(1)'], 'E1013: Argument 1: type mismatch, expected string but got number')
1807 v9.CheckDefFailure(['g:MyVarargsOnly("one", 2)'], 'E1013: Argument 2: type mismatch, expected string but got number')
Bram Moolenaar0b76b422020-04-07 22:05:08 +02001808enddef
1809
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001810def Test_using_var_as_arg()
Bram Moolenaard2939812021-12-30 17:09:05 +00001811 var lines =<< trim END
1812 def Func(x: number)
1813 var x = 234
1814 enddef
1815 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001816 v9.CheckDefFailure(lines, 'E1006:')
Bram Moolenaard2939812021-12-30 17:09:05 +00001817
1818 lines =<< trim END
1819 def Func(Ref: number)
1820 def Ref()
1821 enddef
1822 enddef
1823 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001824 v9.CheckDefFailure(lines, 'E1073:')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001825enddef
1826
Bram Moolenaar62aec932022-01-29 21:45:34 +00001827def s:DictArg(arg: dict<string>)
Bram Moolenaarcb2bdb12020-05-10 22:53:56 +02001828 arg['key'] = 'value'
1829enddef
1830
Bram Moolenaar62aec932022-01-29 21:45:34 +00001831def s:ListArg(arg: list<string>)
Bram Moolenaarcb2bdb12020-05-10 22:53:56 +02001832 arg[0] = 'value'
1833enddef
1834
1835def Test_assign_to_argument()
Bram Moolenaarf5be8cd2020-07-17 20:36:00 +02001836 # works for dict and list
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001837 var d: dict<string> = {}
Bram Moolenaarcb2bdb12020-05-10 22:53:56 +02001838 DictArg(d)
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001839 d['key']->assert_equal('value')
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001840 var l: list<string> = []
Bram Moolenaarcb2bdb12020-05-10 22:53:56 +02001841 ListArg(l)
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001842 l[0]->assert_equal('value')
Bram Moolenaarcb2bdb12020-05-10 22:53:56 +02001843
Bram Moolenaar62aec932022-01-29 21:45:34 +00001844 v9.CheckScriptFailure(['def Func(arg: number)', 'arg = 3', 'enddef', 'defcompile'], 'E1090:')
Bram Moolenaar2d870f82020-12-05 13:41:01 +01001845 delfunc! g:Func
Bram Moolenaarcb2bdb12020-05-10 22:53:56 +02001846enddef
1847
Bram Moolenaarb816dae2020-09-20 22:04:00 +02001848" These argument names are reserved in legacy functions.
Bram Moolenaar62aec932022-01-29 21:45:34 +00001849def s:WithReservedNames(firstline: string, lastline: string): string
Bram Moolenaarb816dae2020-09-20 22:04:00 +02001850 return firstline .. lastline
1851enddef
1852
1853def Test_argument_names()
1854 assert_equal('OK', WithReservedNames('O', 'K'))
1855enddef
1856
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001857def Test_call_func_defined_later()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001858 g:DefinedLater('one')->assert_equal('one')
Bram Moolenaar9bd5d872020-09-06 21:47:48 +02001859 assert_fails('NotDefined("one")', 'E117:', '', 2, 'Test_call_func_defined_later')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001860enddef
1861
Bram Moolenaar1df8b3f2020-04-23 18:13:23 +02001862func DefinedLater(arg)
1863 return a:arg
1864endfunc
1865
1866def Test_call_funcref()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001867 g:SomeFunc('abc')->assert_equal(3)
Bram Moolenaar9bd5d872020-09-06 21:47:48 +02001868 assert_fails('NotAFunc()', 'E117:', '', 2, 'Test_call_funcref') # comment after call
Bram Moolenaar2ef91562021-12-11 16:14:07 +00001869 assert_fails('g:NotAFunc()', 'E1085:', '', 3, 'Test_call_funcref')
Bram Moolenaar2f1980f2020-07-22 19:30:06 +02001870
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001871 var lines =<< trim END
Bram Moolenaar2f1980f2020-07-22 19:30:06 +02001872 vim9script
1873 def RetNumber(): number
1874 return 123
1875 enddef
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001876 var Funcref: func: number = function('RetNumber')
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001877 Funcref()->assert_equal(123)
Bram Moolenaar2f1980f2020-07-22 19:30:06 +02001878 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001879 v9.CheckScriptSuccess(lines)
Bram Moolenaar0f60e802020-07-22 20:16:11 +02001880
1881 lines =<< trim END
1882 vim9script
1883 def RetNumber(): number
1884 return 123
1885 enddef
1886 def Bar(F: func: number): number
1887 return F()
1888 enddef
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001889 var Funcref = function('RetNumber')
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001890 Bar(Funcref)->assert_equal(123)
Bram Moolenaar0f60e802020-07-22 20:16:11 +02001891 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001892 v9.CheckScriptSuccess(lines)
Bram Moolenaarbfba8652020-07-23 20:09:10 +02001893
1894 lines =<< trim END
1895 vim9script
1896 def UseNumber(nr: number)
1897 echo nr
1898 enddef
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001899 var Funcref: func(number) = function('UseNumber')
Bram Moolenaarbfba8652020-07-23 20:09:10 +02001900 Funcref(123)
1901 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001902 v9.CheckScriptSuccess(lines)
Bram Moolenaarb8070e32020-07-23 20:56:04 +02001903
1904 lines =<< trim END
1905 vim9script
1906 def UseNumber(nr: number)
1907 echo nr
1908 enddef
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001909 var Funcref: func(string) = function('UseNumber')
Bram Moolenaarb8070e32020-07-23 20:56:04 +02001910 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001911 v9.CheckScriptFailure(lines, 'E1012: Type mismatch; expected func(string) but got func(number)')
Bram Moolenaar4fc224c2020-07-26 17:56:25 +02001912
1913 lines =<< trim END
1914 vim9script
1915 def EchoNr(nr = 34)
1916 g:echo = nr
1917 enddef
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001918 var Funcref: func(?number) = function('EchoNr')
Bram Moolenaar4fc224c2020-07-26 17:56:25 +02001919 Funcref()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001920 g:echo->assert_equal(34)
Bram Moolenaar4fc224c2020-07-26 17:56:25 +02001921 Funcref(123)
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001922 g:echo->assert_equal(123)
Bram Moolenaar4fc224c2020-07-26 17:56:25 +02001923 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001924 v9.CheckScriptSuccess(lines)
Bram Moolenaarace61322020-07-26 18:16:58 +02001925
1926 lines =<< trim END
1927 vim9script
1928 def EchoList(...l: list<number>)
1929 g:echo = l
1930 enddef
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001931 var Funcref: func(...list<number>) = function('EchoList')
Bram Moolenaarace61322020-07-26 18:16:58 +02001932 Funcref()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001933 g:echo->assert_equal([])
Bram Moolenaarace61322020-07-26 18:16:58 +02001934 Funcref(1, 2, 3)
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001935 g:echo->assert_equal([1, 2, 3])
Bram Moolenaarace61322020-07-26 18:16:58 +02001936 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001937 v9.CheckScriptSuccess(lines)
Bram Moolenaar01865ad2020-07-26 18:33:09 +02001938
1939 lines =<< trim END
1940 vim9script
1941 def OptAndVar(nr: number, opt = 12, ...l: list<number>): number
1942 g:optarg = opt
1943 g:listarg = l
1944 return nr
1945 enddef
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001946 var Funcref: func(number, ?number, ...list<number>): number = function('OptAndVar')
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001947 Funcref(10)->assert_equal(10)
1948 g:optarg->assert_equal(12)
1949 g:listarg->assert_equal([])
Bram Moolenaar01865ad2020-07-26 18:33:09 +02001950
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001951 Funcref(11, 22)->assert_equal(11)
1952 g:optarg->assert_equal(22)
1953 g:listarg->assert_equal([])
Bram Moolenaar01865ad2020-07-26 18:33:09 +02001954
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001955 Funcref(17, 18, 1, 2, 3)->assert_equal(17)
1956 g:optarg->assert_equal(18)
1957 g:listarg->assert_equal([1, 2, 3])
Bram Moolenaar01865ad2020-07-26 18:33:09 +02001958 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001959 v9.CheckScriptSuccess(lines)
Bram Moolenaar1df8b3f2020-04-23 18:13:23 +02001960enddef
1961
1962let SomeFunc = function('len')
1963let NotAFunc = 'text'
1964
Bram Moolenaar99aaf0c2020-04-12 14:39:53 +02001965def CombineFuncrefTypes()
Bram Moolenaarf5be8cd2020-07-17 20:36:00 +02001966 # same arguments, different return type
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001967 var Ref1: func(bool): string
1968 var Ref2: func(bool): number
1969 var Ref3: func(bool): any
Bram Moolenaar99aaf0c2020-04-12 14:39:53 +02001970 Ref3 = g:cond ? Ref1 : Ref2
1971
Bram Moolenaarf5be8cd2020-07-17 20:36:00 +02001972 # different number of arguments
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001973 var Refa1: func(bool): number
1974 var Refa2: func(bool, number): number
1975 var Refa3: func: number
Bram Moolenaar99aaf0c2020-04-12 14:39:53 +02001976 Refa3 = g:cond ? Refa1 : Refa2
1977
Bram Moolenaarf5be8cd2020-07-17 20:36:00 +02001978 # different argument types
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001979 var Refb1: func(bool, string): number
1980 var Refb2: func(string, number): number
1981 var Refb3: func(any, any): number
Bram Moolenaar99aaf0c2020-04-12 14:39:53 +02001982 Refb3 = g:cond ? Refb1 : Refb2
1983enddef
1984
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001985def FuncWithForwardCall()
Bram Moolenaar1df8b3f2020-04-23 18:13:23 +02001986 return g:DefinedEvenLater("yes")
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001987enddef
1988
1989def DefinedEvenLater(arg: string): string
1990 return arg
1991enddef
1992
1993def Test_error_in_nested_function()
Bram Moolenaarf5be8cd2020-07-17 20:36:00 +02001994 # Error in called function requires unwinding the call stack.
Bram Moolenaar848fadd2022-01-30 15:28:30 +00001995 assert_fails('g:FuncWithForwardCall()', 'E1096:', '', 1, 'FuncWithForwardCall')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001996enddef
1997
Bram Moolenaar4bf10062021-12-28 17:23:12 +00001998def Test_nested_function_with_nextcmd()
Bram Moolenaar9c23f9b2021-12-26 14:23:22 +00001999 var lines =<< trim END
2000 vim9script
2001 # Define an outer function
2002 def FirstFunction()
2003 # Define an inner function
2004 def SecondFunction()
2005 # the function has a body, a double free is detected.
2006 AAAAA
2007
2008 # enddef followed by | or } followed by # one or more characters
2009 enddef|BBBB
2010 enddef
2011
2012 # Compile all functions
2013 defcompile
2014 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002015 v9.CheckScriptFailure(lines, 'E1173: Text found after enddef: BBBB')
Bram Moolenaar9c23f9b2021-12-26 14:23:22 +00002016enddef
2017
Bram Moolenaar4bf10062021-12-28 17:23:12 +00002018def Test_nested_function_with_args_split()
2019 var lines =<< trim END
2020 vim9script
2021 def FirstFunction()
2022 def SecondFunction(
2023 )
2024 # had a double free if the right parenthesis of the nested function is
2025 # on the next line
2026
2027 enddef|BBBB
2028 enddef
2029 # Compile all functions
2030 defcompile
2031 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002032 v9.CheckScriptFailure(lines, 'E1173: Text found after enddef: BBBB')
Bram Moolenaar7473a842021-12-28 17:55:26 +00002033
2034 lines =<< trim END
2035 vim9script
2036 def FirstFunction()
2037 func SecondFunction()
2038 endfunc|BBBB
2039 enddef
2040 defcompile
2041 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002042 v9.CheckScriptFailure(lines, 'E1173: Text found after endfunction: BBBB')
Bram Moolenaar4bf10062021-12-28 17:23:12 +00002043enddef
2044
Bram Moolenaar9f1a39a2022-01-08 15:39:39 +00002045def Test_error_in_function_args()
2046 var lines =<< trim END
2047 def FirstFunction()
2048 def SecondFunction(J =
2049 # Nois
2050 # one
2051
2052 enddef|BBBB
2053 enddef
2054 # Compile all functions
2055 defcompile
2056 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002057 v9.CheckScriptFailure(lines, 'E488:')
Bram Moolenaar9f1a39a2022-01-08 15:39:39 +00002058enddef
2059
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002060def Test_return_type_wrong()
Bram Moolenaar62aec932022-01-29 21:45:34 +00002061 v9.CheckScriptFailure([
Bram Moolenaar5a849da2020-08-08 16:47:30 +02002062 'def Func(): number',
2063 'return "a"',
2064 'enddef',
2065 'defcompile'], 'expected number but got string')
Bram Moolenaar2d870f82020-12-05 13:41:01 +01002066 delfunc! g:Func
Bram Moolenaar62aec932022-01-29 21:45:34 +00002067 v9.CheckScriptFailure([
Bram Moolenaar5a849da2020-08-08 16:47:30 +02002068 'def Func(): string',
2069 'return 1',
2070 'enddef',
2071 'defcompile'], 'expected string but got number')
Bram Moolenaar2d870f82020-12-05 13:41:01 +01002072 delfunc! g:Func
Bram Moolenaar62aec932022-01-29 21:45:34 +00002073 v9.CheckScriptFailure([
Bram Moolenaar5a849da2020-08-08 16:47:30 +02002074 'def Func(): void',
2075 'return "a"',
2076 'enddef',
2077 'defcompile'],
2078 'E1096: Returning a value in a function without a return type')
Bram Moolenaar2d870f82020-12-05 13:41:01 +01002079 delfunc! g:Func
Bram Moolenaar62aec932022-01-29 21:45:34 +00002080 v9.CheckScriptFailure([
Bram Moolenaar5a849da2020-08-08 16:47:30 +02002081 'def Func()',
2082 'return "a"',
2083 'enddef',
2084 'defcompile'],
2085 'E1096: Returning a value in a function without a return type')
Bram Moolenaar2d870f82020-12-05 13:41:01 +01002086 delfunc! g:Func
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002087
Bram Moolenaar62aec932022-01-29 21:45:34 +00002088 v9.CheckScriptFailure([
Bram Moolenaar5a849da2020-08-08 16:47:30 +02002089 'def Func(): number',
2090 'return',
2091 'enddef',
2092 'defcompile'], 'E1003:')
Bram Moolenaar2d870f82020-12-05 13:41:01 +01002093 delfunc! g:Func
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002094
Bram Moolenaar62aec932022-01-29 21:45:34 +00002095 v9.CheckScriptFailure([
Bram Moolenaar33ea9fd2021-08-08 19:07:37 +02002096 'def Func():number',
2097 'return 123',
2098 'enddef',
2099 'defcompile'], 'E1069:')
2100 delfunc! g:Func
2101
Bram Moolenaar62aec932022-01-29 21:45:34 +00002102 v9.CheckScriptFailure([
Bram Moolenaar33ea9fd2021-08-08 19:07:37 +02002103 'def Func() :number',
2104 'return 123',
2105 'enddef',
2106 'defcompile'], 'E1059:')
2107 delfunc! g:Func
2108
Bram Moolenaar62aec932022-01-29 21:45:34 +00002109 v9.CheckScriptFailure([
Bram Moolenaar33ea9fd2021-08-08 19:07:37 +02002110 'def Func() : number',
2111 'return 123',
2112 'enddef',
2113 'defcompile'], 'E1059:')
2114 delfunc! g:Func
2115
Bram Moolenaar62aec932022-01-29 21:45:34 +00002116 v9.CheckScriptFailure(['def Func(): list', 'return []', 'enddef'], 'E1008:')
Bram Moolenaar2d870f82020-12-05 13:41:01 +01002117 delfunc! g:Func
Bram Moolenaar62aec932022-01-29 21:45:34 +00002118 v9.CheckScriptFailure(['def Func(): dict', 'return {}', 'enddef'], 'E1008:')
Bram Moolenaar2d870f82020-12-05 13:41:01 +01002119 delfunc! g:Func
Bram Moolenaar62aec932022-01-29 21:45:34 +00002120 v9.CheckScriptFailure(['def Func()', 'return 1'], 'E1057:')
Bram Moolenaar2d870f82020-12-05 13:41:01 +01002121 delfunc! g:Func
Bram Moolenaar5a849da2020-08-08 16:47:30 +02002122
Bram Moolenaar62aec932022-01-29 21:45:34 +00002123 v9.CheckScriptFailure([
Bram Moolenaar5a849da2020-08-08 16:47:30 +02002124 'vim9script',
2125 'def FuncB()',
2126 ' return 123',
2127 'enddef',
2128 'def FuncA()',
2129 ' FuncB()',
2130 'enddef',
2131 'defcompile'], 'E1096:')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002132enddef
2133
2134def Test_arg_type_wrong()
Bram Moolenaar62aec932022-01-29 21:45:34 +00002135 v9.CheckScriptFailure(['def Func3(items: list)', 'echo "a"', 'enddef'], 'E1008: Missing <type>')
2136 v9.CheckScriptFailure(['def Func4(...)', 'echo "a"', 'enddef'], 'E1055: Missing name after ...')
2137 v9.CheckScriptFailure(['def Func5(items:string)', 'echo "a"'], 'E1069:')
2138 v9.CheckScriptFailure(['def Func5(items)', 'echo "a"'], 'E1077:')
2139 v9.CheckScriptFailure(['def Func6(...x:list<number>)', 'echo "a"', 'enddef'], 'E1069:')
2140 v9.CheckScriptFailure(['def Func7(...x: int)', 'echo "a"', 'enddef'], 'E1010:')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002141enddef
2142
Bram Moolenaar86cdb8a2021-04-06 19:01:03 +02002143def Test_white_space_before_comma()
2144 var lines =<< trim END
2145 vim9script
2146 def Func(a: number , b: number)
2147 enddef
2148 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002149 v9.CheckScriptFailure(lines, 'E1068:')
Yegappan Lakshmanan611728f2021-05-24 15:15:47 +02002150 call assert_fails('vim9cmd echo stridx("a" .. "b" , "a")', 'E1068:')
Bram Moolenaar86cdb8a2021-04-06 19:01:03 +02002151enddef
2152
Bram Moolenaar608d78f2021-03-06 22:33:12 +01002153def Test_white_space_after_comma()
2154 var lines =<< trim END
2155 vim9script
2156 def Func(a: number,b: number)
2157 enddef
2158 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002159 v9.CheckScriptFailure(lines, 'E1069:')
Bram Moolenaar608d78f2021-03-06 22:33:12 +01002160
2161 # OK in legacy function
2162 lines =<< trim END
2163 vim9script
2164 func Func(a,b)
2165 endfunc
2166 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002167 v9.CheckScriptSuccess(lines)
Bram Moolenaar608d78f2021-03-06 22:33:12 +01002168enddef
2169
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002170def Test_vim9script_call()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002171 var lines =<< trim END
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002172 vim9script
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002173 var name = ''
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002174 def MyFunc(arg: string)
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002175 name = arg
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002176 enddef
2177 MyFunc('foobar')
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002178 name->assert_equal('foobar')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002179
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002180 var str = 'barfoo'
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002181 str->MyFunc()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002182 name->assert_equal('barfoo')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002183
Bram Moolenaar67979662020-06-20 22:50:47 +02002184 g:value = 'value'
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002185 g:value->MyFunc()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002186 name->assert_equal('value')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002187
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002188 var listvar = []
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002189 def ListFunc(arg: list<number>)
2190 listvar = arg
2191 enddef
2192 [1, 2, 3]->ListFunc()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002193 listvar->assert_equal([1, 2, 3])
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002194
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002195 var dictvar = {}
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002196 def DictFunc(arg: dict<number>)
2197 dictvar = arg
2198 enddef
Bram Moolenaare0de1712020-12-02 17:36:54 +01002199 {a: 1, b: 2}->DictFunc()
2200 dictvar->assert_equal({a: 1, b: 2})
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002201 def CompiledDict()
Bram Moolenaare0de1712020-12-02 17:36:54 +01002202 {a: 3, b: 4}->DictFunc()
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002203 enddef
2204 CompiledDict()
Bram Moolenaare0de1712020-12-02 17:36:54 +01002205 dictvar->assert_equal({a: 3, b: 4})
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002206
Bram Moolenaare0de1712020-12-02 17:36:54 +01002207 {a: 3, b: 4}->DictFunc()
2208 dictvar->assert_equal({a: 3, b: 4})
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002209
2210 ('text')->MyFunc()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002211 name->assert_equal('text')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002212 ("some")->MyFunc()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002213 name->assert_equal('some')
Bram Moolenaare6b53242020-07-01 17:28:33 +02002214
Bram Moolenaar13e12b82020-07-24 18:47:22 +02002215 # line starting with single quote is not a mark
Bram Moolenaar10409562020-07-29 20:00:38 +02002216 # line starting with double quote can be a method call
Bram Moolenaar3d48e252020-07-15 14:15:52 +02002217 'asdfasdf'->MyFunc()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002218 name->assert_equal('asdfasdf')
Bram Moolenaar10409562020-07-29 20:00:38 +02002219 "xyz"->MyFunc()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002220 name->assert_equal('xyz')
Bram Moolenaar3d48e252020-07-15 14:15:52 +02002221
2222 def UseString()
2223 'xyork'->MyFunc()
2224 enddef
2225 UseString()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002226 name->assert_equal('xyork')
Bram Moolenaar3d48e252020-07-15 14:15:52 +02002227
Bram Moolenaar10409562020-07-29 20:00:38 +02002228 def UseString2()
2229 "knife"->MyFunc()
2230 enddef
2231 UseString2()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002232 name->assert_equal('knife')
Bram Moolenaar10409562020-07-29 20:00:38 +02002233
Bram Moolenaar13e12b82020-07-24 18:47:22 +02002234 # prepending a colon makes it a mark
2235 new
2236 setline(1, ['aaa', 'bbb', 'ccc'])
2237 normal! 3Gmt1G
2238 :'t
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002239 getcurpos()[1]->assert_equal(3)
Bram Moolenaar13e12b82020-07-24 18:47:22 +02002240 bwipe!
2241
Bram Moolenaare6b53242020-07-01 17:28:33 +02002242 MyFunc(
2243 'continued'
2244 )
2245 assert_equal('continued',
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002246 name
Bram Moolenaare6b53242020-07-01 17:28:33 +02002247 )
2248
2249 call MyFunc(
2250 'more'
2251 ..
2252 'lines'
2253 )
2254 assert_equal(
2255 'morelines',
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002256 name)
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002257 END
2258 writefile(lines, 'Xcall.vim')
2259 source Xcall.vim
2260 delete('Xcall.vim')
2261enddef
2262
2263def Test_vim9script_call_fail_decl()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002264 var lines =<< trim END
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002265 vim9script
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002266 var name = ''
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002267 def MyFunc(arg: string)
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002268 var name = 123
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002269 enddef
Bram Moolenaar822ba242020-05-24 23:00:18 +02002270 defcompile
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002271 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002272 v9.CheckScriptFailure(lines, 'E1054:')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002273enddef
2274
Bram Moolenaar65b95452020-07-19 14:03:09 +02002275def Test_vim9script_call_fail_type()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002276 var lines =<< trim END
Bram Moolenaar65b95452020-07-19 14:03:09 +02002277 vim9script
2278 def MyFunc(arg: string)
2279 echo arg
2280 enddef
2281 MyFunc(1234)
2282 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002283 v9.CheckScriptFailure(lines, 'E1013: Argument 1: type mismatch, expected string but got number')
Bram Moolenaar65b95452020-07-19 14:03:09 +02002284enddef
2285
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002286def Test_vim9script_call_fail_const()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002287 var lines =<< trim END
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002288 vim9script
2289 const var = ''
2290 def MyFunc(arg: string)
2291 var = 'asdf'
2292 enddef
Bram Moolenaar822ba242020-05-24 23:00:18 +02002293 defcompile
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002294 END
2295 writefile(lines, 'Xcall_const.vim')
Bram Moolenaar9bd5d872020-09-06 21:47:48 +02002296 assert_fails('source Xcall_const.vim', 'E46:', '', 1, 'MyFunc')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002297 delete('Xcall_const.vim')
Bram Moolenaar3bdc90b2020-12-22 20:35:40 +01002298
2299 lines =<< trim END
2300 const g:Aconst = 77
2301 def Change()
2302 # comment
2303 g:Aconst = 99
2304 enddef
2305 call Change()
2306 unlet g:Aconst
2307 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002308 v9.CheckScriptFailure(lines, 'E741: Value is locked: Aconst', 2)
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002309enddef
2310
2311" Test that inside :function a Python function can be defined, :def is not
2312" recognized.
2313func Test_function_python()
2314 CheckFeature python3
Bram Moolenaar727345e2020-09-27 23:33:59 +02002315 let py = 'python3'
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002316 execute py "<< EOF"
2317def do_something():
2318 return 1
2319EOF
2320endfunc
2321
2322def Test_delfunc()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002323 var lines =<< trim END
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002324 vim9script
Bram Moolenaar4c17ad92020-04-27 22:47:51 +02002325 def g:GoneSoon()
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002326 echo 'hello'
2327 enddef
2328
2329 def CallGoneSoon()
Bram Moolenaar62aec932022-01-29 21:45:34 +00002330 g:GoneSoon()
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002331 enddef
Bram Moolenaar822ba242020-05-24 23:00:18 +02002332 defcompile
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002333
Bram Moolenaar4c17ad92020-04-27 22:47:51 +02002334 delfunc g:GoneSoon
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002335 CallGoneSoon()
2336 END
2337 writefile(lines, 'XToDelFunc')
Bram Moolenaar9bd5d872020-09-06 21:47:48 +02002338 assert_fails('so XToDelFunc', 'E933:', '', 1, 'CallGoneSoon')
2339 assert_fails('so XToDelFunc', 'E933:', '', 1, 'CallGoneSoon')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002340
2341 delete('XToDelFunc')
2342enddef
2343
Bram Moolenaar7509ad82021-12-14 18:14:37 +00002344func Test_free_dict_while_in_funcstack()
2345 " relies on the sleep command
2346 CheckUnix
2347 call Run_Test_free_dict_while_in_funcstack()
2348endfunc
2349
2350def Run_Test_free_dict_while_in_funcstack()
Bram Moolenaar7509ad82021-12-14 18:14:37 +00002351 # this was freeing the TermRun() default argument dictionary while it was
2352 # still referenced in a funcstack_T
2353 var lines =<< trim END
2354 vim9script
2355
2356 &updatetime = 400
2357 def TermRun(_ = {})
2358 def Post()
2359 enddef
2360 def Exec()
2361 term_start('sleep 1', {
2362 term_finish: 'close',
2363 exit_cb: (_, _) => Post(),
2364 })
2365 enddef
2366 Exec()
2367 enddef
2368 nnoremap <F4> <Cmd>call <SID>TermRun()<CR>
2369 timer_start(100, (_) => feedkeys("\<F4>"))
2370 timer_start(1000, (_) => feedkeys("\<F4>"))
2371 sleep 1500m
2372 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002373 v9.CheckScriptSuccess(lines)
Bram Moolenaar7509ad82021-12-14 18:14:37 +00002374 nunmap <F4>
2375 set updatetime&
2376enddef
2377
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002378def Test_redef_failure()
Bram Moolenaard2c61702020-09-06 15:58:36 +02002379 writefile(['def Func0(): string', 'return "Func0"', 'enddef'], 'Xdef')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002380 so Xdef
Bram Moolenaard2c61702020-09-06 15:58:36 +02002381 writefile(['def Func1(): string', 'return "Func1"', 'enddef'], 'Xdef')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002382 so Xdef
Bram Moolenaard2c61702020-09-06 15:58:36 +02002383 writefile(['def! Func0(): string', 'enddef', 'defcompile'], 'Xdef')
Bram Moolenaar9bd5d872020-09-06 21:47:48 +02002384 assert_fails('so Xdef', 'E1027:', '', 1, 'Func0')
Bram Moolenaard2c61702020-09-06 15:58:36 +02002385 writefile(['def Func2(): string', 'return "Func2"', 'enddef'], 'Xdef')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002386 so Xdef
Bram Moolenaard2c61702020-09-06 15:58:36 +02002387 delete('Xdef')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002388
Bram Moolenaar701cc6c2021-04-10 13:33:48 +02002389 assert_fails('g:Func0()', 'E1091:')
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002390 g:Func1()->assert_equal('Func1')
2391 g:Func2()->assert_equal('Func2')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002392
2393 delfunc! Func0
2394 delfunc! Func1
2395 delfunc! Func2
2396enddef
2397
Bram Moolenaarf93c7fe2020-04-23 22:16:53 +02002398def Test_vim9script_func()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002399 var lines =<< trim END
Bram Moolenaarf93c7fe2020-04-23 22:16:53 +02002400 vim9script
2401 func Func(arg)
2402 echo a:arg
2403 endfunc
2404 Func('text')
2405 END
2406 writefile(lines, 'XVim9Func')
2407 so XVim9Func
2408
2409 delete('XVim9Func')
2410enddef
2411
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002412let s:funcResult = 0
2413
2414def FuncNoArgNoRet()
Bram Moolenaar53900992020-08-22 19:02:02 +02002415 s:funcResult = 11
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002416enddef
2417
2418def FuncNoArgRetNumber(): number
Bram Moolenaar53900992020-08-22 19:02:02 +02002419 s:funcResult = 22
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002420 return 1234
2421enddef
2422
Bram Moolenaarec5929d2020-04-07 20:53:39 +02002423def FuncNoArgRetString(): string
Bram Moolenaar53900992020-08-22 19:02:02 +02002424 s:funcResult = 45
Bram Moolenaarec5929d2020-04-07 20:53:39 +02002425 return 'text'
2426enddef
2427
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002428def FuncOneArgNoRet(arg: number)
Bram Moolenaar53900992020-08-22 19:02:02 +02002429 s:funcResult = arg
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002430enddef
2431
2432def FuncOneArgRetNumber(arg: number): number
Bram Moolenaar53900992020-08-22 19:02:02 +02002433 s:funcResult = arg
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002434 return arg
2435enddef
2436
Bram Moolenaar08938ee2020-04-11 23:17:17 +02002437def FuncTwoArgNoRet(one: bool, two: number)
Bram Moolenaar53900992020-08-22 19:02:02 +02002438 s:funcResult = two
Bram Moolenaar08938ee2020-04-11 23:17:17 +02002439enddef
2440
Bram Moolenaar62aec932022-01-29 21:45:34 +00002441def s:FuncOneArgRetString(arg: string): string
Bram Moolenaarec5929d2020-04-07 20:53:39 +02002442 return arg
2443enddef
2444
Bram Moolenaar62aec932022-01-29 21:45:34 +00002445def s:FuncOneArgRetAny(arg: any): any
Bram Moolenaar89228602020-04-05 22:14:54 +02002446 return arg
2447enddef
2448
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002449def Test_func_type()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002450 var Ref1: func()
Bram Moolenaar53900992020-08-22 19:02:02 +02002451 s:funcResult = 0
Bram Moolenaar62aec932022-01-29 21:45:34 +00002452 Ref1 = g:FuncNoArgNoRet
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002453 Ref1()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002454 s:funcResult->assert_equal(11)
Bram Moolenaar4c683752020-04-05 21:38:23 +02002455
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002456 var Ref2: func
Bram Moolenaar53900992020-08-22 19:02:02 +02002457 s:funcResult = 0
Bram Moolenaar62aec932022-01-29 21:45:34 +00002458 Ref2 = g:FuncNoArgNoRet
Bram Moolenaar4c683752020-04-05 21:38:23 +02002459 Ref2()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002460 s:funcResult->assert_equal(11)
Bram Moolenaar4c683752020-04-05 21:38:23 +02002461
Bram Moolenaar53900992020-08-22 19:02:02 +02002462 s:funcResult = 0
Bram Moolenaar62aec932022-01-29 21:45:34 +00002463 Ref2 = g:FuncOneArgNoRet
Bram Moolenaar4c683752020-04-05 21:38:23 +02002464 Ref2(12)
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002465 s:funcResult->assert_equal(12)
Bram Moolenaar4c683752020-04-05 21:38:23 +02002466
Bram Moolenaar53900992020-08-22 19:02:02 +02002467 s:funcResult = 0
Bram Moolenaar62aec932022-01-29 21:45:34 +00002468 Ref2 = g:FuncNoArgRetNumber
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002469 Ref2()->assert_equal(1234)
2470 s:funcResult->assert_equal(22)
Bram Moolenaar4c683752020-04-05 21:38:23 +02002471
Bram Moolenaar53900992020-08-22 19:02:02 +02002472 s:funcResult = 0
Bram Moolenaar848fadd2022-01-30 15:28:30 +00002473 Ref2 = g:FuncOneArgRetNumber
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002474 Ref2(13)->assert_equal(13)
2475 s:funcResult->assert_equal(13)
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002476enddef
2477
Bram Moolenaar9978d472020-07-05 16:01:56 +02002478def Test_repeat_return_type()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002479 var res = 0
Bram Moolenaar9978d472020-07-05 16:01:56 +02002480 for n in repeat([1], 3)
2481 res += n
2482 endfor
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002483 res->assert_equal(3)
Bram Moolenaarfce82b32020-07-05 16:07:21 +02002484
2485 res = 0
2486 for n in add([1, 2], 3)
2487 res += n
2488 endfor
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002489 res->assert_equal(6)
Bram Moolenaar9978d472020-07-05 16:01:56 +02002490enddef
2491
Bram Moolenaar846178a2020-07-05 17:04:13 +02002492def Test_argv_return_type()
2493 next fileone filetwo
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002494 var res = ''
Bram Moolenaar846178a2020-07-05 17:04:13 +02002495 for name in argv()
2496 res ..= name
2497 endfor
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002498 res->assert_equal('fileonefiletwo')
Bram Moolenaar846178a2020-07-05 17:04:13 +02002499enddef
2500
Bram Moolenaarec5929d2020-04-07 20:53:39 +02002501def Test_func_type_part()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002502 var RefVoid: func: void
Bram Moolenaar62aec932022-01-29 21:45:34 +00002503 RefVoid = g:FuncNoArgNoRet
2504 RefVoid = g:FuncOneArgNoRet
2505 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 +00002506 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 +02002507
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002508 var RefAny: func(): any
Bram Moolenaar62aec932022-01-29 21:45:34 +00002509 RefAny = g:FuncNoArgRetNumber
Bram Moolenaar848fadd2022-01-30 15:28:30 +00002510 RefAny = g:FuncNoArgRetString
Bram Moolenaar62aec932022-01-29 21:45:34 +00002511 v9.CheckDefFailure(['var RefAny: func(): any', 'RefAny = g:FuncNoArgNoRet'], 'E1012: Type mismatch; expected func(): any but got func()')
2512 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 +02002513
Bram Moolenaar6abd3dc2020-10-04 14:17:32 +02002514 var RefAnyNoArgs: func: any = RefAny
2515
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002516 var RefNr: func: number
Bram Moolenaar62aec932022-01-29 21:45:34 +00002517 RefNr = g:FuncNoArgRetNumber
Bram Moolenaar848fadd2022-01-30 15:28:30 +00002518 RefNr = g:FuncOneArgRetNumber
Bram Moolenaar62aec932022-01-29 21:45:34 +00002519 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 +00002520 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 +02002521
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002522 var RefStr: func: string
Bram Moolenaar848fadd2022-01-30 15:28:30 +00002523 RefStr = g:FuncNoArgRetString
Bram Moolenaarec5929d2020-04-07 20:53:39 +02002524 RefStr = FuncOneArgRetString
Bram Moolenaar62aec932022-01-29 21:45:34 +00002525 v9.CheckDefFailure(['var RefStr: func: string', 'RefStr = g:FuncNoArgNoRet'], 'E1012: Type mismatch; expected func(...): string but got func()')
2526 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 +02002527enddef
2528
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002529def Test_func_type_fails()
Bram Moolenaar62aec932022-01-29 21:45:34 +00002530 v9.CheckDefFailure(['var ref1: func()'], 'E704:')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002531
Bram Moolenaar62aec932022-01-29 21:45:34 +00002532 v9.CheckDefFailure(['var Ref1: func()', 'Ref1 = g:FuncNoArgRetNumber'], 'E1012: Type mismatch; expected func() but got func(): number')
2533 v9.CheckDefFailure(['var Ref1: func()', 'Ref1 = g:FuncOneArgNoRet'], 'E1012: Type mismatch; expected func() but got func(number)')
Bram Moolenaar848fadd2022-01-30 15:28:30 +00002534 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 +00002535 v9.CheckDefFailure(['var Ref1: func(bool)', 'Ref1 = g:FuncTwoArgNoRet'], 'E1012: Type mismatch; expected func(bool) but got func(bool, number)')
2536 v9.CheckDefFailure(['var Ref1: func(?bool)', 'Ref1 = g:FuncTwoArgNoRet'], 'E1012: Type mismatch; expected func(?bool) but got func(bool, number)')
2537 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 +02002538
Bram Moolenaar62aec932022-01-29 21:45:34 +00002539 v9.CheckDefFailure(['var RefWrong: func(string ,number)'], 'E1068:')
2540 v9.CheckDefFailure(['var RefWrong: func(string,number)'], 'E1069:')
2541 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:')
2542 v9.CheckDefFailure(['var RefWrong: func(bool):string'], 'E1069:')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002543enddef
2544
Bram Moolenaar89228602020-04-05 22:14:54 +02002545def Test_func_return_type()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002546 var nr: number
Bram Moolenaar62aec932022-01-29 21:45:34 +00002547 nr = g:FuncNoArgRetNumber()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002548 nr->assert_equal(1234)
Bram Moolenaar89228602020-04-05 22:14:54 +02002549
2550 nr = FuncOneArgRetAny(122)
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002551 nr->assert_equal(122)
Bram Moolenaar89228602020-04-05 22:14:54 +02002552
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002553 var str: string
Bram Moolenaar89228602020-04-05 22:14:54 +02002554 str = FuncOneArgRetAny('yes')
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002555 str->assert_equal('yes')
Bram Moolenaar89228602020-04-05 22:14:54 +02002556
Bram Moolenaar62aec932022-01-29 21:45:34 +00002557 v9.CheckDefFailure(['var str: string', 'str = g:FuncNoArgRetNumber()'], 'E1012: Type mismatch; expected string but got number')
Bram Moolenaar89228602020-04-05 22:14:54 +02002558enddef
2559
Bram Moolenaar6abd3dc2020-10-04 14:17:32 +02002560def Test_func_common_type()
2561 def FuncOne(n: number): number
2562 return n
2563 enddef
2564 def FuncTwo(s: string): number
2565 return len(s)
2566 enddef
2567 def FuncThree(n: number, s: string): number
2568 return n + len(s)
2569 enddef
2570 var list = [FuncOne, FuncTwo, FuncThree]
2571 assert_equal(8, list[0](8))
2572 assert_equal(4, list[1]('word'))
2573 assert_equal(7, list[2](3, 'word'))
2574enddef
2575
Bram Moolenaar62aec932022-01-29 21:45:34 +00002576def s:MultiLine(
Bram Moolenaar5e774c72020-04-12 21:53:00 +02002577 arg1: string,
2578 arg2 = 1234,
2579 ...rest: list<string>
2580 ): string
2581 return arg1 .. arg2 .. join(rest, '-')
2582enddef
2583
Bram Moolenaar2c330432020-04-13 14:41:35 +02002584def MultiLineComment(
2585 arg1: string, # comment
2586 arg2 = 1234, # comment
2587 ...rest: list<string> # comment
2588 ): string # comment
2589 return arg1 .. arg2 .. join(rest, '-')
2590enddef
2591
Bram Moolenaar5e774c72020-04-12 21:53:00 +02002592def Test_multiline()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002593 MultiLine('text')->assert_equal('text1234')
2594 MultiLine('text', 777)->assert_equal('text777')
2595 MultiLine('text', 777, 'one')->assert_equal('text777one')
2596 MultiLine('text', 777, 'one', 'two')->assert_equal('text777one-two')
Bram Moolenaar5e774c72020-04-12 21:53:00 +02002597enddef
2598
Bram Moolenaar23e03252020-04-12 22:22:31 +02002599func Test_multiline_not_vim9()
Bram Moolenaar62aec932022-01-29 21:45:34 +00002600 call s:MultiLine('text')->assert_equal('text1234')
2601 call s:MultiLine('text', 777)->assert_equal('text777')
2602 call s:MultiLine('text', 777, 'one')->assert_equal('text777one')
2603 call s:MultiLine('text', 777, 'one', 'two')->assert_equal('text777one-two')
Bram Moolenaar23e03252020-04-12 22:22:31 +02002604endfunc
2605
Bram Moolenaar5e774c72020-04-12 21:53:00 +02002606
Bram Moolenaaree4e0c12020-04-06 21:35:05 +02002607" When using CheckScriptFailure() for the below test, E1010 is generated instead
2608" of E1056.
2609func Test_E1056_1059()
2610 let caught_1056 = 0
2611 try
2612 def F():
2613 return 1
2614 enddef
2615 catch /E1056:/
2616 let caught_1056 = 1
2617 endtry
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002618 eval caught_1056->assert_equal(1)
Bram Moolenaaree4e0c12020-04-06 21:35:05 +02002619
2620 let caught_1059 = 0
2621 try
2622 def F5(items : list)
2623 echo 'a'
2624 enddef
2625 catch /E1059:/
2626 let caught_1059 = 1
2627 endtry
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002628 eval caught_1059->assert_equal(1)
Bram Moolenaaree4e0c12020-04-06 21:35:05 +02002629endfunc
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002630
Bram Moolenaar015f4262020-05-05 21:25:22 +02002631func DelMe()
2632 echo 'DelMe'
2633endfunc
2634
Bram Moolenaarbf8feb52020-08-08 14:26:31 +02002635def Test_error_reporting()
2636 # comment lines at the start of the function
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002637 var lines =<< trim END
Bram Moolenaarbf8feb52020-08-08 14:26:31 +02002638 " comment
2639 def Func()
2640 # comment
2641 # comment
2642 invalid
2643 enddef
2644 defcompile
2645 END
Bram Moolenaar08052222020-09-14 17:04:31 +02002646 writefile(lines, 'Xdef')
Bram Moolenaarbf8feb52020-08-08 14:26:31 +02002647 try
2648 source Xdef
Bram Moolenaar7517ffd2020-08-14 18:35:07 +02002649 assert_report('should have failed')
Bram Moolenaarbf8feb52020-08-08 14:26:31 +02002650 catch /E476:/
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002651 v:exception->assert_match('Invalid command: invalid')
2652 v:throwpoint->assert_match(', line 3$')
Bram Moolenaarbf8feb52020-08-08 14:26:31 +02002653 endtry
Bram Moolenaar2d870f82020-12-05 13:41:01 +01002654 delfunc! g:Func
Bram Moolenaarbf8feb52020-08-08 14:26:31 +02002655
2656 # comment lines after the start of the function
2657 lines =<< trim END
2658 " comment
2659 def Func()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002660 var x = 1234
Bram Moolenaarbf8feb52020-08-08 14:26:31 +02002661 # comment
2662 # comment
2663 invalid
2664 enddef
2665 defcompile
2666 END
Bram Moolenaar08052222020-09-14 17:04:31 +02002667 writefile(lines, 'Xdef')
Bram Moolenaarbf8feb52020-08-08 14:26:31 +02002668 try
2669 source Xdef
Bram Moolenaar7517ffd2020-08-14 18:35:07 +02002670 assert_report('should have failed')
Bram Moolenaarbf8feb52020-08-08 14:26:31 +02002671 catch /E476:/
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002672 v:exception->assert_match('Invalid command: invalid')
2673 v:throwpoint->assert_match(', line 4$')
Bram Moolenaarbf8feb52020-08-08 14:26:31 +02002674 endtry
Bram Moolenaar2d870f82020-12-05 13:41:01 +01002675 delfunc! g:Func
Bram Moolenaarbf8feb52020-08-08 14:26:31 +02002676
Bram Moolenaar7517ffd2020-08-14 18:35:07 +02002677 lines =<< trim END
2678 vim9script
2679 def Func()
Bram Moolenaare0de1712020-12-02 17:36:54 +01002680 var db = {foo: 1, bar: 2}
Bram Moolenaar7517ffd2020-08-14 18:35:07 +02002681 # comment
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002682 var x = db.asdf
Bram Moolenaar7517ffd2020-08-14 18:35:07 +02002683 enddef
2684 defcompile
2685 Func()
2686 END
Bram Moolenaar08052222020-09-14 17:04:31 +02002687 writefile(lines, 'Xdef')
Bram Moolenaar7517ffd2020-08-14 18:35:07 +02002688 try
2689 source Xdef
2690 assert_report('should have failed')
2691 catch /E716:/
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002692 v:throwpoint->assert_match('_Func, line 3$')
Bram Moolenaar7517ffd2020-08-14 18:35:07 +02002693 endtry
Bram Moolenaar2d870f82020-12-05 13:41:01 +01002694 delfunc! g:Func
Bram Moolenaar7517ffd2020-08-14 18:35:07 +02002695
Bram Moolenaar08052222020-09-14 17:04:31 +02002696 delete('Xdef')
Bram Moolenaarbf8feb52020-08-08 14:26:31 +02002697enddef
2698
Bram Moolenaar015f4262020-05-05 21:25:22 +02002699def Test_deleted_function()
Bram Moolenaar62aec932022-01-29 21:45:34 +00002700 v9.CheckDefExecFailure([
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002701 'var RefMe: func = function("g:DelMe")',
Bram Moolenaar015f4262020-05-05 21:25:22 +02002702 'delfunc g:DelMe',
2703 'echo RefMe()'], 'E117:')
2704enddef
2705
2706def Test_unknown_function()
Bram Moolenaar62aec932022-01-29 21:45:34 +00002707 v9.CheckDefExecFailure([
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002708 'var Ref: func = function("NotExist")',
Bram Moolenaar9b7bf9e2020-07-11 22:14:59 +02002709 'delfunc g:NotExist'], 'E700:')
Bram Moolenaar015f4262020-05-05 21:25:22 +02002710enddef
2711
Bram Moolenaar62aec932022-01-29 21:45:34 +00002712def s:RefFunc(Ref: func(any): any): string
Bram Moolenaarc8cd2b32020-05-01 19:29:08 +02002713 return Ref('more')
2714enddef
2715
2716def Test_closure_simple()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002717 var local = 'some '
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01002718 RefFunc((s) => local .. s)->assert_equal('some more')
Bram Moolenaarc8cd2b32020-05-01 19:29:08 +02002719enddef
2720
Bram Moolenaar62aec932022-01-29 21:45:34 +00002721def s:MakeRef()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002722 var local = 'some '
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01002723 g:Ref = (s) => local .. s
Bram Moolenaarbf67ea12020-05-02 17:52:42 +02002724enddef
2725
2726def Test_closure_ref_after_return()
2727 MakeRef()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002728 g:Ref('thing')->assert_equal('some thing')
Bram Moolenaarbf67ea12020-05-02 17:52:42 +02002729 unlet g:Ref
2730enddef
2731
Bram Moolenaar62aec932022-01-29 21:45:34 +00002732def s:MakeTwoRefs()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002733 var local = ['some']
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01002734 g:Extend = (s) => local->add(s)
2735 g:Read = () => local
Bram Moolenaar5adc55c2020-05-02 23:12:58 +02002736enddef
2737
2738def Test_closure_two_refs()
2739 MakeTwoRefs()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002740 join(g:Read(), ' ')->assert_equal('some')
Bram Moolenaar5adc55c2020-05-02 23:12:58 +02002741 g:Extend('more')
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002742 join(g:Read(), ' ')->assert_equal('some more')
Bram Moolenaar5adc55c2020-05-02 23:12:58 +02002743 g:Extend('even')
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002744 join(g:Read(), ' ')->assert_equal('some more even')
Bram Moolenaar5adc55c2020-05-02 23:12:58 +02002745
2746 unlet g:Extend
2747 unlet g:Read
2748enddef
2749
Bram Moolenaar62aec932022-01-29 21:45:34 +00002750def s:ReadRef(Ref: func(): list<string>): string
Bram Moolenaar5adc55c2020-05-02 23:12:58 +02002751 return join(Ref(), ' ')
2752enddef
2753
Bram Moolenaar62aec932022-01-29 21:45:34 +00002754def s:ExtendRef(Ref: func(string): list<string>, add: string)
Bram Moolenaar5adc55c2020-05-02 23:12:58 +02002755 Ref(add)
2756enddef
2757
2758def Test_closure_two_indirect_refs()
Bram Moolenaarf7779c62020-05-03 15:38:16 +02002759 MakeTwoRefs()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002760 ReadRef(g:Read)->assert_equal('some')
Bram Moolenaar5adc55c2020-05-02 23:12:58 +02002761 ExtendRef(g:Extend, 'more')
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002762 ReadRef(g:Read)->assert_equal('some more')
Bram Moolenaar5adc55c2020-05-02 23:12:58 +02002763 ExtendRef(g:Extend, 'even')
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002764 ReadRef(g:Read)->assert_equal('some more even')
Bram Moolenaar5adc55c2020-05-02 23:12:58 +02002765
2766 unlet g:Extend
2767 unlet g:Read
2768enddef
Bram Moolenaarbf67ea12020-05-02 17:52:42 +02002769
Bram Moolenaar62aec932022-01-29 21:45:34 +00002770def s:MakeArgRefs(theArg: string)
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002771 var local = 'loc_val'
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01002772 g:UseArg = (s) => theArg .. '/' .. local .. '/' .. s
Bram Moolenaar2fd4cd72020-05-03 22:30:49 +02002773enddef
2774
Bram Moolenaar62aec932022-01-29 21:45:34 +00002775def s:MakeArgRefsVarargs(theArg: string, ...rest: list<string>)
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002776 var local = 'the_loc'
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01002777 g:UseVararg = (s) => theArg .. '/' .. local .. '/' .. s .. '/' .. join(rest)
Bram Moolenaar2fd4cd72020-05-03 22:30:49 +02002778enddef
2779
2780def Test_closure_using_argument()
2781 MakeArgRefs('arg_val')
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002782 g:UseArg('call_val')->assert_equal('arg_val/loc_val/call_val')
Bram Moolenaar2fd4cd72020-05-03 22:30:49 +02002783
2784 MakeArgRefsVarargs('arg_val', 'one', 'two')
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002785 g:UseVararg('call_val')->assert_equal('arg_val/the_loc/call_val/one two')
Bram Moolenaar2fd4cd72020-05-03 22:30:49 +02002786
2787 unlet g:UseArg
2788 unlet g:UseVararg
Bram Moolenaar44ec21c2021-02-12 21:50:57 +01002789
2790 var lines =<< trim END
2791 vim9script
2792 def Test(Fun: func(number): number): list<number>
2793 return map([1, 2, 3], (_, i) => Fun(i))
2794 enddef
2795 def Inc(nr: number): number
2796 return nr + 2
2797 enddef
2798 assert_equal([3, 4, 5], Test(Inc))
2799 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002800 v9.CheckScriptSuccess(lines)
Bram Moolenaar2fd4cd72020-05-03 22:30:49 +02002801enddef
2802
Bram Moolenaar62aec932022-01-29 21:45:34 +00002803def s:MakeGetAndAppendRefs()
Bram Moolenaar85d5e2b2020-10-10 14:13:01 +02002804 var local = 'a'
2805
2806 def Append(arg: string)
2807 local ..= arg
2808 enddef
2809 g:Append = Append
2810
2811 def Get(): string
2812 return local
2813 enddef
2814 g:Get = Get
2815enddef
2816
2817def Test_closure_append_get()
2818 MakeGetAndAppendRefs()
2819 g:Get()->assert_equal('a')
2820 g:Append('-b')
2821 g:Get()->assert_equal('a-b')
2822 g:Append('-c')
2823 g:Get()->assert_equal('a-b-c')
2824
2825 unlet g:Append
2826 unlet g:Get
2827enddef
Bram Moolenaarb68b3462020-05-06 21:06:30 +02002828
Bram Moolenaar04b12692020-05-04 23:24:44 +02002829def Test_nested_closure()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002830 var local = 'text'
Bram Moolenaar04b12692020-05-04 23:24:44 +02002831 def Closure(arg: string): string
2832 return local .. arg
2833 enddef
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002834 Closure('!!!')->assert_equal('text!!!')
Bram Moolenaar04b12692020-05-04 23:24:44 +02002835enddef
2836
Bram Moolenaar62aec932022-01-29 21:45:34 +00002837func s:GetResult(Ref)
Bram Moolenaar6f5b6df2020-05-16 21:20:12 +02002838 return a:Ref('some')
2839endfunc
2840
2841def Test_call_closure_not_compiled()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002842 var text = 'text'
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01002843 g:Ref = (s) => s .. text
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002844 GetResult(g:Ref)->assert_equal('sometext')
Bram Moolenaar6f5b6df2020-05-16 21:20:12 +02002845enddef
2846
Bram Moolenaar7cbfaa52020-09-18 21:25:32 +02002847def Test_double_closure_fails()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002848 var lines =<< trim END
Bram Moolenaar7cbfaa52020-09-18 21:25:32 +02002849 vim9script
2850 def Func()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002851 var name = 0
2852 for i in range(2)
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01002853 timer_start(0, () => name)
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002854 endfor
Bram Moolenaar7cbfaa52020-09-18 21:25:32 +02002855 enddef
2856 Func()
2857 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002858 v9.CheckScriptSuccess(lines)
Bram Moolenaar7cbfaa52020-09-18 21:25:32 +02002859enddef
2860
Bram Moolenaar85d5e2b2020-10-10 14:13:01 +02002861def Test_nested_closure_used()
2862 var lines =<< trim END
2863 vim9script
2864 def Func()
2865 var x = 'hello'
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01002866 var Closure = () => x
2867 g:Myclosure = () => Closure()
Bram Moolenaar85d5e2b2020-10-10 14:13:01 +02002868 enddef
2869 Func()
2870 assert_equal('hello', g:Myclosure())
2871 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002872 v9.CheckScriptSuccess(lines)
Bram Moolenaar85d5e2b2020-10-10 14:13:01 +02002873enddef
Bram Moolenaar0876c782020-10-07 19:08:04 +02002874
Bram Moolenaarc70bdab2020-09-26 19:59:38 +02002875def Test_nested_closure_fails()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002876 var lines =<< trim END
Bram Moolenaarc70bdab2020-09-26 19:59:38 +02002877 vim9script
2878 def FuncA()
2879 FuncB(0)
2880 enddef
2881 def FuncB(n: number): list<string>
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01002882 return map([0], (_, v) => n)
Bram Moolenaarc70bdab2020-09-26 19:59:38 +02002883 enddef
2884 FuncA()
2885 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002886 v9.CheckScriptFailure(lines, 'E1012:')
Bram Moolenaarc70bdab2020-09-26 19:59:38 +02002887enddef
2888
Bram Moolenaarf112f302020-12-20 17:47:52 +01002889def Test_global_closure()
2890 var lines =<< trim END
2891 vim9script
2892 def ReverseEveryNLines(n: number, line1: number, line2: number)
2893 var mods = 'sil keepj keepp lockm '
2894 var range = ':' .. line1 .. ',' .. line2
2895 def g:Offset(): number
2896 var offset = (line('.') - line1 + 1) % n
2897 return offset != 0 ? offset : n
2898 enddef
2899 exe mods .. range .. 'g/^/exe "m .-" .. g:Offset()'
2900 enddef
2901
2902 new
2903 repeat(['aaa', 'bbb', 'ccc'], 3)->setline(1)
2904 ReverseEveryNLines(3, 1, 9)
2905 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002906 v9.CheckScriptSuccess(lines)
Bram Moolenaarf112f302020-12-20 17:47:52 +01002907 var expected = repeat(['ccc', 'bbb', 'aaa'], 3)
2908 assert_equal(expected, getline(1, 9))
2909 bwipe!
2910enddef
2911
Bram Moolenaarcd45ed02020-12-22 17:35:54 +01002912def Test_global_closure_called_directly()
2913 var lines =<< trim END
2914 vim9script
2915 def Outer()
2916 var x = 1
2917 def g:Inner()
2918 var y = x
2919 x += 1
2920 assert_equal(1, y)
2921 enddef
2922 g:Inner()
2923 assert_equal(2, x)
2924 enddef
2925 Outer()
2926 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002927 v9.CheckScriptSuccess(lines)
Bram Moolenaarcd45ed02020-12-22 17:35:54 +01002928 delfunc g:Inner
2929enddef
2930
Bram Moolenaar69c76172021-12-02 16:38:52 +00002931def Test_closure_called_from_legacy()
2932 var lines =<< trim END
2933 vim9script
2934 def Func()
2935 var outer = 'foo'
2936 var F = () => {
2937 outer = 'bar'
2938 }
2939 execute printf('call %s()', string(F))
2940 enddef
2941 Func()
2942 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002943 v9.CheckScriptFailure(lines, 'E1248')
Bram Moolenaar69c76172021-12-02 16:38:52 +00002944enddef
2945
Bram Moolenaar34c54eb2020-11-25 19:15:19 +01002946def Test_failure_in_called_function()
2947 # this was using the frame index as the return value
2948 var lines =<< trim END
2949 vim9script
2950 au TerminalWinOpen * eval [][0]
2951 def PopupTerm(a: any)
2952 # make sure typvals on stack are string
2953 ['a', 'b', 'c', 'd', 'e', 'f', 'g']->join()
2954 FireEvent()
2955 enddef
2956 def FireEvent()
2957 do TerminalWinOpen
2958 enddef
2959 # use try/catch to make eval fail
2960 try
2961 call PopupTerm(0)
2962 catch
2963 endtry
2964 au! TerminalWinOpen
2965 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002966 v9.CheckScriptSuccess(lines)
Bram Moolenaar34c54eb2020-11-25 19:15:19 +01002967enddef
2968
Bram Moolenaar5366e1a2020-10-01 13:01:34 +02002969def Test_nested_lambda()
2970 var lines =<< trim END
2971 vim9script
2972 def Func()
2973 var x = 4
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01002974 var Lambda1 = () => 7
2975 var Lambda2 = () => [Lambda1(), x]
Bram Moolenaar5366e1a2020-10-01 13:01:34 +02002976 var res = Lambda2()
2977 assert_equal([7, 4], res)
2978 enddef
2979 Func()
2980 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002981 v9.CheckScriptSuccess(lines)
Bram Moolenaar5366e1a2020-10-01 13:01:34 +02002982enddef
2983
Bram Moolenaarc04f2a42021-06-09 19:30:03 +02002984def Test_double_nested_lambda()
2985 var lines =<< trim END
2986 vim9script
2987 def F(head: string): func(string): func(string): string
2988 return (sep: string): func(string): string => ((tail: string): string => {
2989 return head .. sep .. tail
2990 })
2991 enddef
2992 assert_equal('hello-there', F('hello')('-')('there'))
2993 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002994 v9.CheckScriptSuccess(lines)
Bram Moolenaarc04f2a42021-06-09 19:30:03 +02002995enddef
2996
Bram Moolenaar074f84c2021-05-18 11:47:44 +02002997def Test_nested_inline_lambda()
Bram Moolenaar074f84c2021-05-18 11:47:44 +02002998 var lines =<< trim END
2999 vim9script
3000 def F(text: string): func(string): func(string): string
3001 return (arg: string): func(string): string => ((sep: string): string => {
Bram Moolenaar23e2e112021-08-03 21:16:18 +02003002 return sep .. arg .. text
Bram Moolenaar074f84c2021-05-18 11:47:44 +02003003 })
3004 enddef
Bram Moolenaar23e2e112021-08-03 21:16:18 +02003005 assert_equal('--there++', F('++')('there')('--'))
Bram Moolenaar074f84c2021-05-18 11:47:44 +02003006 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003007 v9.CheckScriptSuccess(lines)
Bram Moolenaar5245beb2021-07-15 22:03:50 +02003008
3009 lines =<< trim END
3010 vim9script
3011 echo range(4)->mapnew((_, v) => {
3012 return range(v) ->mapnew((_, s) => {
3013 return string(s)
3014 })
3015 })
3016 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003017 v9.CheckScriptSuccess(lines)
Bram Moolenaarc6ba2f92021-07-18 13:42:29 +02003018
3019 lines =<< trim END
3020 vim9script
3021
Bram Moolenaara749a422022-02-12 19:52:25 +00003022 def Func()
Bram Moolenaarc6ba2f92021-07-18 13:42:29 +02003023 range(10)
3024 ->mapnew((_, _) => ({
3025 key: range(10)->mapnew((_, _) => {
3026 return ' '
3027 }),
3028 }))
3029 enddef
3030
3031 defcomp
3032 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003033 v9.CheckScriptSuccess(lines)
Bram Moolenaar074f84c2021-05-18 11:47:44 +02003034enddef
3035
Bram Moolenaar52bf81c2020-11-17 18:50:44 +01003036def Shadowed(): list<number>
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01003037 var FuncList: list<func: number> = [() => 42]
Bram Moolenaar75ab91f2021-01-10 22:42:50 +01003038 return FuncList->mapnew((_, Shadowed) => Shadowed())
Bram Moolenaar52bf81c2020-11-17 18:50:44 +01003039enddef
3040
3041def Test_lambda_arg_shadows_func()
Bram Moolenaar62aec932022-01-29 21:45:34 +00003042 assert_equal([42], g:Shadowed())
Bram Moolenaar52bf81c2020-11-17 18:50:44 +01003043enddef
3044
Bram Moolenaar21dc8f12022-03-16 17:54:17 +00003045def Test_compiling_referenced_func_no_shadow()
3046 var lines =<< trim END
3047 vim9script
3048
3049 def InitializeReply(lspserver: dict<any>)
3050 enddef
3051
3052 def ProcessReply(lspserver: dict<any>)
3053 var lsp_reply_handlers: dict<func> =
3054 { 'initialize': InitializeReply }
3055 lsp_reply_handlers['initialize'](lspserver)
3056 enddef
3057
3058 call ProcessReply({})
3059 END
3060 v9.CheckScriptSuccess(lines)
3061enddef
3062
Bram Moolenaar62aec932022-01-29 21:45:34 +00003063def s:Line_continuation_in_def(dir: string = ''): string
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02003064 var path: string = empty(dir)
3065 \ ? 'empty'
3066 \ : 'full'
3067 return path
Bram Moolenaaracd4c5e2020-06-22 19:39:03 +02003068enddef
3069
3070def Test_line_continuation_in_def()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02003071 Line_continuation_in_def('.')->assert_equal('full')
Bram Moolenaaracd4c5e2020-06-22 19:39:03 +02003072enddef
3073
Bram Moolenaar2ea95b62020-11-19 21:47:56 +01003074def Test_script_var_in_lambda()
3075 var lines =<< trim END
3076 vim9script
3077 var script = 'test'
Bram Moolenaarbb8a7ce2021-04-10 20:10:26 +02003078 assert_equal(['test'], map(['one'], (_, _) => script))
Bram Moolenaar2ea95b62020-11-19 21:47:56 +01003079 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003080 v9.CheckScriptSuccess(lines)
Bram Moolenaar2ea95b62020-11-19 21:47:56 +01003081enddef
3082
Bram Moolenaar62aec932022-01-29 21:45:34 +00003083def s:Line_continuation_in_lambda(): list<string>
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02003084 var x = range(97, 100)
Bram Moolenaar75ab91f2021-01-10 22:42:50 +01003085 ->mapnew((_, v) => nr2char(v)
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01003086 ->toupper())
Bram Moolenaar7a4b8982020-07-08 17:36:21 +02003087 ->reverse()
3088 return x
3089enddef
3090
3091def Test_line_continuation_in_lambda()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02003092 Line_continuation_in_lambda()->assert_equal(['D', 'C', 'B', 'A'])
Bram Moolenaarf898f7c2021-01-16 18:09:52 +01003093
3094 var lines =<< trim END
3095 vim9script
3096 var res = [{n: 1, m: 2, s: 'xxx'}]
3097 ->mapnew((_, v: dict<any>): string => printf('%d:%d:%s',
3098 v.n,
3099 v.m,
3100 substitute(v.s, '.*', 'yyy', '')
3101 ))
3102 assert_equal(['1:2:yyy'], res)
3103 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003104 v9.CheckScriptSuccess(lines)
Bram Moolenaar7a4b8982020-07-08 17:36:21 +02003105enddef
3106
Bram Moolenaarb6571982021-01-08 22:24:19 +01003107def Test_list_lambda()
3108 timer_start(1000, (_) => 0)
3109 var body = execute(timer_info()[0].callback
3110 ->string()
3111 ->substitute("('", ' ', '')
3112 ->substitute("')", '', '')
3113 ->substitute('function\zs', ' ', ''))
Bram Moolenaar767034c2021-04-09 17:24:52 +02003114 assert_match('def <lambda>\d\+(_: any): number\n1 return 0\n enddef', body)
Bram Moolenaarb6571982021-01-08 22:24:19 +01003115enddef
3116
Bram Moolenaar3c77b6a2021-07-25 18:07:00 +02003117def Test_lambda_block_variable()
Bram Moolenaar88421d62021-07-24 14:14:52 +02003118 var lines =<< trim END
3119 vim9script
3120 var flist: list<func>
3121 for i in range(10)
3122 var inloop = i
3123 flist[i] = () => inloop
3124 endfor
3125 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003126 v9.CheckScriptSuccess(lines)
Bram Moolenaar88421d62021-07-24 14:14:52 +02003127
3128 lines =<< trim END
3129 vim9script
3130 if true
3131 var outloop = 5
3132 var flist: list<func>
3133 for i in range(10)
3134 flist[i] = () => outloop
3135 endfor
3136 endif
3137 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003138 v9.CheckScriptSuccess(lines)
Bram Moolenaar88421d62021-07-24 14:14:52 +02003139
3140 lines =<< trim END
3141 vim9script
3142 if true
3143 var outloop = 5
3144 endif
3145 var flist: list<func>
3146 for i in range(10)
3147 flist[i] = () => outloop
3148 endfor
3149 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003150 v9.CheckScriptFailure(lines, 'E1001: Variable not found: outloop', 1)
Bram Moolenaar3c77b6a2021-07-25 18:07:00 +02003151
3152 lines =<< trim END
3153 vim9script
3154 for i in range(10)
3155 var Ref = () => 0
3156 endfor
3157 assert_equal(0, ((i) => 0)(0))
3158 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003159 v9.CheckScriptSuccess(lines)
Bram Moolenaar88421d62021-07-24 14:14:52 +02003160enddef
3161
Bram Moolenaar96cf4ba2021-04-24 14:15:41 +02003162def Test_legacy_lambda()
3163 legacy echo {x -> 'hello ' .. x}('foo')
Bram Moolenaardc4c2302021-04-25 13:54:42 +02003164
Bram Moolenaar96cf4ba2021-04-24 14:15:41 +02003165 var lines =<< trim END
3166 echo {x -> 'hello ' .. x}('foo')
3167 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003168 v9.CheckDefAndScriptFailure(lines, 'E720:')
Bram Moolenaardc4c2302021-04-25 13:54:42 +02003169
3170 lines =<< trim END
3171 vim9script
3172 def Func()
3173 echo (() => 'no error')()
3174 enddef
3175 legacy call s:Func()
3176 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003177 v9.CheckScriptSuccess(lines)
Bram Moolenaar96cf4ba2021-04-24 14:15:41 +02003178enddef
3179
Bram Moolenaarce024c32021-06-26 13:00:49 +02003180def Test_legacy()
3181 var lines =<< trim END
3182 vim9script
3183 func g:LegacyFunction()
3184 let g:legacyvar = 1
3185 endfunc
3186 def Testit()
3187 legacy call g:LegacyFunction()
3188 enddef
3189 Testit()
3190 assert_equal(1, g:legacyvar)
3191 unlet g:legacyvar
3192 delfunc g:LegacyFunction
3193 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003194 v9.CheckScriptSuccess(lines)
Bram Moolenaarce024c32021-06-26 13:00:49 +02003195enddef
3196
Bram Moolenaarc3cb1c92021-06-02 16:47:53 +02003197def Test_legacy_errors()
3198 for cmd in ['if', 'elseif', 'else', 'endif',
3199 'for', 'endfor', 'continue', 'break',
3200 'while', 'endwhile',
3201 'try', 'catch', 'finally', 'endtry']
Bram Moolenaar62aec932022-01-29 21:45:34 +00003202 v9.CheckDefFailure(['legacy ' .. cmd .. ' expr'], 'E1189:')
Bram Moolenaarc3cb1c92021-06-02 16:47:53 +02003203 endfor
3204enddef
3205
Bram Moolenaarb1b6f4d2021-09-13 18:25:54 +02003206def Test_call_legacy_with_dict()
3207 var lines =<< trim END
3208 vim9script
3209 func Legacy() dict
3210 let g:result = self.value
3211 endfunc
3212 def TestDirect()
3213 var d = {value: 'yes', func: Legacy}
3214 d.func()
3215 enddef
3216 TestDirect()
3217 assert_equal('yes', g:result)
3218 unlet g:result
3219
3220 def TestIndirect()
3221 var d = {value: 'foo', func: Legacy}
3222 var Fi = d.func
3223 Fi()
3224 enddef
3225 TestIndirect()
3226 assert_equal('foo', g:result)
3227 unlet g:result
3228
3229 var d = {value: 'bar', func: Legacy}
3230 d.func()
3231 assert_equal('bar', g:result)
3232 unlet g:result
3233 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003234 v9.CheckScriptSuccess(lines)
Bram Moolenaarb1b6f4d2021-09-13 18:25:54 +02003235enddef
3236
Bram Moolenaar62aec932022-01-29 21:45:34 +00003237def s:DoFilterThis(a: string): list<string>
Bram Moolenaarab360522021-01-10 14:02:28 +01003238 # closure nested inside another closure using argument
3239 var Filter = (l) => filter(l, (_, v) => stridx(v, a) == 0)
3240 return ['x', 'y', 'a', 'x2', 'c']->Filter()
3241enddef
3242
3243def Test_nested_closure_using_argument()
3244 assert_equal(['x', 'x2'], DoFilterThis('x'))
3245enddef
3246
Bram Moolenaar0186e582021-01-10 18:33:11 +01003247def Test_triple_nested_closure()
3248 var what = 'x'
3249 var Match = (val: string, cmp: string): bool => stridx(val, cmp) == 0
3250 var Filter = (l) => filter(l, (_, v) => Match(v, what))
3251 assert_equal(['x', 'x2'], ['x', 'y', 'a', 'x2', 'c']->Filter())
3252enddef
3253
Bram Moolenaar8f510af2020-07-05 18:48:23 +02003254func Test_silent_echo()
Bram Moolenaar47e7d702020-07-05 18:18:42 +02003255 CheckScreendump
Bram Moolenaar3b309f12021-12-13 18:19:55 +00003256 call Run_Test_silent_echo()
3257endfunc
Bram Moolenaar47e7d702020-07-05 18:18:42 +02003258
Bram Moolenaar3b309f12021-12-13 18:19:55 +00003259def Run_Test_silent_echo()
3260 var lines =<< trim END
Bram Moolenaar47e7d702020-07-05 18:18:42 +02003261 vim9script
3262 def EchoNothing()
3263 silent echo ''
3264 enddef
3265 defcompile
3266 END
Bram Moolenaar3b309f12021-12-13 18:19:55 +00003267 writefile(lines, 'XTest_silent_echo')
Bram Moolenaar47e7d702020-07-05 18:18:42 +02003268
Bram Moolenaar3b309f12021-12-13 18:19:55 +00003269 # Check that the balloon shows up after a mouse move
Bram Moolenaar62aec932022-01-29 21:45:34 +00003270 var buf = g:RunVimInTerminal('-S XTest_silent_echo', {'rows': 6})
Bram Moolenaar3b309f12021-12-13 18:19:55 +00003271 term_sendkeys(buf, ":abc")
Bram Moolenaar62aec932022-01-29 21:45:34 +00003272 g:VerifyScreenDump(buf, 'Test_vim9_silent_echo', {})
Bram Moolenaar47e7d702020-07-05 18:18:42 +02003273
Bram Moolenaar3b309f12021-12-13 18:19:55 +00003274 # clean up
Bram Moolenaar62aec932022-01-29 21:45:34 +00003275 g:StopVimInTerminal(buf)
Bram Moolenaar3b309f12021-12-13 18:19:55 +00003276 delete('XTest_silent_echo')
3277enddef
Bram Moolenaar47e7d702020-07-05 18:18:42 +02003278
Bram Moolenaar171fb922020-10-28 16:54:47 +01003279def SilentlyError()
3280 execute('silent! invalid')
3281 g:did_it = 'yes'
3282enddef
3283
Bram Moolenaar62aec932022-01-29 21:45:34 +00003284func s:UserError()
Bram Moolenaar28ee8922020-10-28 20:20:00 +01003285 silent! invalid
3286endfunc
3287
3288def SilentlyUserError()
3289 UserError()
3290 g:did_it = 'yes'
3291enddef
Bram Moolenaar171fb922020-10-28 16:54:47 +01003292
3293" This can't be a :def function, because the assert would not be reached.
Bram Moolenaar171fb922020-10-28 16:54:47 +01003294func Test_ignore_silent_error()
3295 let g:did_it = 'no'
3296 call SilentlyError()
3297 call assert_equal('yes', g:did_it)
3298
Bram Moolenaar28ee8922020-10-28 20:20:00 +01003299 let g:did_it = 'no'
3300 call SilentlyUserError()
3301 call assert_equal('yes', g:did_it)
Bram Moolenaar171fb922020-10-28 16:54:47 +01003302
3303 unlet g:did_it
3304endfunc
3305
Bram Moolenaarcd030c42020-10-30 21:49:40 +01003306def Test_ignore_silent_error_in_filter()
3307 var lines =<< trim END
3308 vim9script
3309 def Filter(winid: number, key: string): bool
3310 if key == 'o'
3311 silent! eval [][0]
3312 return true
3313 endif
3314 return popup_filter_menu(winid, key)
3315 enddef
3316
Bram Moolenaare0de1712020-12-02 17:36:54 +01003317 popup_create('popup', {filter: Filter})
Bram Moolenaarcd030c42020-10-30 21:49:40 +01003318 feedkeys("o\r", 'xnt')
3319 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003320 v9.CheckScriptSuccess(lines)
Bram Moolenaarcd030c42020-10-30 21:49:40 +01003321enddef
3322
Bram Moolenaar62aec932022-01-29 21:45:34 +00003323def s:Fibonacci(n: number): number
Bram Moolenaar4b9bd692020-09-05 21:57:53 +02003324 if n < 2
3325 return n
3326 else
3327 return Fibonacci(n - 1) + Fibonacci(n - 2)
3328 endif
3329enddef
3330
Bram Moolenaar985116a2020-07-12 17:31:09 +02003331def Test_recursive_call()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02003332 Fibonacci(20)->assert_equal(6765)
Bram Moolenaar985116a2020-07-12 17:31:09 +02003333enddef
3334
Bram Moolenaar62aec932022-01-29 21:45:34 +00003335def s:TreeWalk(dir: string): list<any>
Bram Moolenaar75ab91f2021-01-10 22:42:50 +01003336 return readdir(dir)->mapnew((_, val) =>
Bram Moolenaar08f7a412020-07-13 20:41:08 +02003337 fnamemodify(dir .. '/' .. val, ':p')->isdirectory()
Bram Moolenaar2bede172020-11-19 18:53:18 +01003338 ? {[val]: TreeWalk(dir .. '/' .. val)}
Bram Moolenaar08f7a412020-07-13 20:41:08 +02003339 : val
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01003340 )
Bram Moolenaar08f7a412020-07-13 20:41:08 +02003341enddef
3342
3343def Test_closure_in_map()
3344 mkdir('XclosureDir/tdir', 'p')
3345 writefile(['111'], 'XclosureDir/file1')
3346 writefile(['222'], 'XclosureDir/file2')
3347 writefile(['333'], 'XclosureDir/tdir/file3')
3348
Bram Moolenaare0de1712020-12-02 17:36:54 +01003349 TreeWalk('XclosureDir')->assert_equal(['file1', 'file2', {tdir: ['file3']}])
Bram Moolenaar08f7a412020-07-13 20:41:08 +02003350
3351 delete('XclosureDir', 'rf')
3352enddef
3353
Bram Moolenaar7b5d5442020-10-04 13:42:34 +02003354def Test_invalid_function_name()
3355 var lines =<< trim END
3356 vim9script
3357 def s: list<string>
3358 END
Bram Moolenaara749a422022-02-12 19:52:25 +00003359 v9.CheckScriptFailure(lines, 'E1268:')
Bram Moolenaar7b5d5442020-10-04 13:42:34 +02003360
3361 lines =<< trim END
3362 vim9script
3363 def g: list<string>
3364 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003365 v9.CheckScriptFailure(lines, 'E129:')
Bram Moolenaar7b5d5442020-10-04 13:42:34 +02003366
3367 lines =<< trim END
3368 vim9script
3369 def <SID>: list<string>
3370 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003371 v9.CheckScriptFailure(lines, 'E884:')
Bram Moolenaar7b5d5442020-10-04 13:42:34 +02003372
3373 lines =<< trim END
3374 vim9script
3375 def F list<string>
3376 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003377 v9.CheckScriptFailure(lines, 'E488:')
Bram Moolenaar7b5d5442020-10-04 13:42:34 +02003378enddef
3379
Bram Moolenaara90afb92020-07-15 22:38:56 +02003380def Test_partial_call()
Bram Moolenaarf78da4f2021-08-01 15:40:31 +02003381 var lines =<< trim END
3382 var Xsetlist: func
3383 Xsetlist = function('setloclist', [0])
3384 Xsetlist([], ' ', {title: 'test'})
3385 getloclist(0, {title: 1})->assert_equal({title: 'test'})
Bram Moolenaara90afb92020-07-15 22:38:56 +02003386
Bram Moolenaarf78da4f2021-08-01 15:40:31 +02003387 Xsetlist = function('setloclist', [0, [], ' '])
3388 Xsetlist({title: 'test'})
3389 getloclist(0, {title: 1})->assert_equal({title: 'test'})
Bram Moolenaara90afb92020-07-15 22:38:56 +02003390
Bram Moolenaarf78da4f2021-08-01 15:40:31 +02003391 Xsetlist = function('setqflist')
3392 Xsetlist([], ' ', {title: 'test'})
3393 getqflist({title: 1})->assert_equal({title: 'test'})
Bram Moolenaara90afb92020-07-15 22:38:56 +02003394
Bram Moolenaarf78da4f2021-08-01 15:40:31 +02003395 Xsetlist = function('setqflist', [[], ' '])
3396 Xsetlist({title: 'test'})
3397 getqflist({title: 1})->assert_equal({title: 'test'})
Bram Moolenaar6abd3dc2020-10-04 14:17:32 +02003398
Bram Moolenaarf78da4f2021-08-01 15:40:31 +02003399 var Len: func: number = function('len', ['word'])
3400 assert_equal(4, Len())
3401
3402 var RepeatFunc = function('repeat', ['o'])
3403 assert_equal('ooooo', RepeatFunc(5))
3404 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003405 v9.CheckDefAndScriptSuccess(lines)
Bram Moolenaarc66f6452021-08-19 21:08:30 +02003406
3407 lines =<< trim END
3408 vim9script
3409 def Foo(Parser: any)
3410 enddef
3411 var Expr: func(dict<any>): dict<any>
3412 const Call = Foo(Expr)
3413 END
Bram Moolenaar8acb9cc2022-03-08 13:18:55 +00003414 v9.CheckScriptFailure(lines, 'E1031:')
Bram Moolenaara90afb92020-07-15 22:38:56 +02003415enddef
3416
Bram Moolenaarcd1cda22022-02-16 21:48:25 +00003417def Test_partial_double_nested()
3418 var idx = 123
3419 var Get = () => idx
3420 var Ref = function(Get, [])
3421 var RefRef = function(Ref, [])
3422 assert_equal(123, RefRef())
3423enddef
3424
Bram Moolenaar673bcb12022-03-08 16:52:24 +00003425def Test_partial_null_function()
3426 var lines =<< trim END
3427 var d: dict<func> = {f: null_function}
3428 var Ref = d.f
Bram Moolenaared0c62e2022-03-08 19:43:55 +00003429 assert_equal('func(...): unknown', typename(Ref))
Bram Moolenaar673bcb12022-03-08 16:52:24 +00003430 END
3431 v9.CheckDefAndScriptSuccess(lines)
3432enddef
3433
Bram Moolenaarfe1bfc92022-02-06 13:55:03 +00003434" Using "idx" from a legacy global function does not work.
3435" This caused a crash when called from legacy context.
3436func Test_partial_call_fails()
3437 let lines =<< trim END
3438 vim9script
3439
3440 var l = ['a', 'b', 'c']
3441 def Iter(container: any): any
3442 var idx = -1
3443 var obj = {state: container}
Bram Moolenaarf681cfb2022-02-07 20:30:57 +00003444 def g:NextItem__(self: dict<any>): any
Bram Moolenaarfe1bfc92022-02-06 13:55:03 +00003445 ++idx
3446 return self.state[idx]
3447 enddef
Bram Moolenaarf681cfb2022-02-07 20:30:57 +00003448 obj.__next__ = function('g:NextItem__', [obj])
Bram Moolenaarfe1bfc92022-02-06 13:55:03 +00003449 return obj
3450 enddef
3451
3452 var it = Iter(l)
3453 echo it.__next__()
3454 END
3455 call writefile(lines, 'XpartialCall')
3456 try
3457 source XpartialCall
3458 catch /E1248:/
3459 endtry
3460 call delete('XpartialCall')
3461endfunc
3462
Bram Moolenaar2dd0a2c2020-08-08 15:10:27 +02003463def Test_cmd_modifier()
3464 tab echo '0'
Bram Moolenaar62aec932022-01-29 21:45:34 +00003465 v9.CheckDefFailure(['5tab echo 3'], 'E16:')
Bram Moolenaar2dd0a2c2020-08-08 15:10:27 +02003466enddef
3467
3468def Test_restore_modifiers()
3469 # check that when compiling a :def function command modifiers are not messed
3470 # up.
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02003471 var lines =<< trim END
Bram Moolenaar2dd0a2c2020-08-08 15:10:27 +02003472 vim9script
3473 set eventignore=
3474 autocmd QuickFixCmdPost * copen
3475 def AutocmdsDisabled()
Bram Moolenaarc3235272021-07-10 19:42:03 +02003476 eval 1 + 2
Bram Moolenaar2dd0a2c2020-08-08 15:10:27 +02003477 enddef
3478 func Func()
3479 noautocmd call s:AutocmdsDisabled()
3480 let g:ei_after = &eventignore
3481 endfunc
3482 Func()
3483 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003484 v9.CheckScriptSuccess(lines)
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02003485 g:ei_after->assert_equal('')
Bram Moolenaar2dd0a2c2020-08-08 15:10:27 +02003486enddef
3487
Bram Moolenaardfa3d552020-09-10 22:05:08 +02003488def StackTop()
Bram Moolenaarc3235272021-07-10 19:42:03 +02003489 eval 1 + 2
3490 eval 2 + 3
Bram Moolenaardfa3d552020-09-10 22:05:08 +02003491 # call not on fourth line
Bram Moolenaar62aec932022-01-29 21:45:34 +00003492 g:StackBot()
Bram Moolenaardfa3d552020-09-10 22:05:08 +02003493enddef
3494
3495def StackBot()
3496 # throw an error
3497 eval [][0]
3498enddef
3499
3500def Test_callstack_def()
3501 try
Bram Moolenaar62aec932022-01-29 21:45:34 +00003502 g:StackTop()
Bram Moolenaardfa3d552020-09-10 22:05:08 +02003503 catch
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02003504 v:throwpoint->assert_match('Test_callstack_def\[2\]..StackTop\[4\]..StackBot, line 2')
Bram Moolenaardfa3d552020-09-10 22:05:08 +02003505 endtry
3506enddef
3507
Bram Moolenaare8211a32020-10-09 22:04:29 +02003508" Re-using spot for variable used in block
3509def Test_block_scoped_var()
3510 var lines =<< trim END
3511 vim9script
3512 def Func()
3513 var x = ['a', 'b', 'c']
3514 if 1
3515 var y = 'x'
Bram Moolenaarbb8a7ce2021-04-10 20:10:26 +02003516 map(x, (_, _) => y)
Bram Moolenaare8211a32020-10-09 22:04:29 +02003517 endif
3518 var z = x
3519 assert_equal(['x', 'x', 'x'], z)
3520 enddef
3521 Func()
3522 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003523 v9.CheckScriptSuccess(lines)
Bram Moolenaare8211a32020-10-09 22:04:29 +02003524enddef
3525
Bram Moolenaareeece9e2020-11-20 19:26:48 +01003526def Test_reset_did_emsg()
3527 var lines =<< trim END
3528 @s = 'blah'
3529 au BufWinLeave * #
3530 def Func()
3531 var winid = popup_create('popup', {})
3532 exe '*s'
3533 popup_close(winid)
3534 enddef
3535 Func()
3536 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003537 v9.CheckScriptFailure(lines, 'E492:', 8)
Bram Moolenaar2d870f82020-12-05 13:41:01 +01003538 delfunc! g:Func
Bram Moolenaareeece9e2020-11-20 19:26:48 +01003539enddef
3540
Bram Moolenaar57f799e2020-12-12 20:42:19 +01003541def Test_did_emsg_reset()
3542 # executing an autocommand resets did_emsg, this should not result in a
3543 # builtin function considered failing
3544 var lines =<< trim END
3545 vim9script
3546 au BufWinLeave * #
3547 def Func()
Bram Moolenaar767034c2021-04-09 17:24:52 +02003548 popup_menu('', {callback: (a, b) => popup_create('', {})->popup_close()})
Bram Moolenaar57f799e2020-12-12 20:42:19 +01003549 eval [][0]
3550 enddef
3551 nno <F3> <cmd>call <sid>Func()<cr>
3552 feedkeys("\<F3>\e", 'xt')
3553 END
3554 writefile(lines, 'XemsgReset')
3555 assert_fails('so XemsgReset', ['E684:', 'E684:'], lines, 2)
3556 delete('XemsgReset')
3557 nunmap <F3>
3558 au! BufWinLeave
3559enddef
3560
Bram Moolenaar56602ba2020-12-05 21:22:08 +01003561def Test_abort_with_silent_call()
3562 var lines =<< trim END
3563 vim9script
3564 g:result = 'none'
3565 def Func()
3566 g:result += 3
3567 g:result = 'yes'
3568 enddef
3569 # error is silenced, but function aborts on error
3570 silent! Func()
3571 assert_equal('none', g:result)
3572 unlet g:result
3573 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003574 v9.CheckScriptSuccess(lines)
Bram Moolenaar56602ba2020-12-05 21:22:08 +01003575enddef
3576
Bram Moolenaarf665e972020-12-05 19:17:16 +01003577def Test_continues_with_silent_error()
3578 var lines =<< trim END
3579 vim9script
3580 g:result = 'none'
3581 def Func()
3582 silent! g:result += 3
3583 g:result = 'yes'
3584 enddef
3585 # error is silenced, function does not abort
3586 Func()
3587 assert_equal('yes', g:result)
3588 unlet g:result
3589 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003590 v9.CheckScriptSuccess(lines)
Bram Moolenaarf665e972020-12-05 19:17:16 +01003591enddef
3592
Bram Moolenaaraf0df472020-12-02 20:51:22 +01003593def Test_abort_even_with_silent()
3594 var lines =<< trim END
3595 vim9script
3596 g:result = 'none'
3597 def Func()
3598 eval {-> ''}() .. '' .. {}['X']
3599 g:result = 'yes'
3600 enddef
Bram Moolenaarf665e972020-12-05 19:17:16 +01003601 silent! Func()
Bram Moolenaaraf0df472020-12-02 20:51:22 +01003602 assert_equal('none', g:result)
Bram Moolenaar4029cab2020-12-05 18:13:27 +01003603 unlet g:result
3604 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003605 v9.CheckScriptSuccess(lines)
Bram Moolenaar4029cab2020-12-05 18:13:27 +01003606enddef
3607
Bram Moolenaarf665e972020-12-05 19:17:16 +01003608def Test_cmdmod_silent_restored()
3609 var lines =<< trim END
3610 vim9script
3611 def Func()
3612 g:result = 'none'
3613 silent! g:result += 3
3614 g:result = 'none'
3615 g:result += 3
3616 enddef
3617 Func()
3618 END
3619 # can't use CheckScriptFailure, it ignores the :silent!
3620 var fname = 'Xdefsilent'
3621 writefile(lines, fname)
3622 var caught = 'no'
3623 try
3624 exe 'source ' .. fname
3625 catch /E1030:/
3626 caught = 'yes'
3627 assert_match('Func, line 4', v:throwpoint)
3628 endtry
3629 assert_equal('yes', caught)
3630 delete(fname)
3631enddef
3632
Bram Moolenaar2fecb532021-03-24 22:00:56 +01003633def Test_cmdmod_silent_nested()
3634 var lines =<< trim END
3635 vim9script
3636 var result = ''
3637
3638 def Error()
3639 result ..= 'Eb'
3640 eval [][0]
3641 result ..= 'Ea'
3642 enddef
3643
3644 def Crash()
3645 result ..= 'Cb'
3646 sil! Error()
3647 result ..= 'Ca'
3648 enddef
3649
3650 Crash()
3651 assert_equal('CbEbEaCa', result)
3652 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003653 v9.CheckScriptSuccess(lines)
Bram Moolenaar2fecb532021-03-24 22:00:56 +01003654enddef
3655
Bram Moolenaar4029cab2020-12-05 18:13:27 +01003656def Test_dict_member_with_silent()
3657 var lines =<< trim END
3658 vim9script
3659 g:result = 'none'
3660 var d: dict<any>
3661 def Func()
3662 try
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01003663 g:result = map([], (_, v) => ({}[v]))->join() .. d['']
Bram Moolenaar4029cab2020-12-05 18:13:27 +01003664 catch
3665 endtry
3666 enddef
3667 silent! Func()
3668 assert_equal('0', g:result)
3669 unlet g:result
Bram Moolenaaraf0df472020-12-02 20:51:22 +01003670 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003671 v9.CheckScriptSuccess(lines)
Bram Moolenaaraf0df472020-12-02 20:51:22 +01003672enddef
3673
Bram Moolenaarf9041332021-01-21 19:41:16 +01003674def Test_skip_cmds_with_silent()
3675 var lines =<< trim END
3676 vim9script
3677
3678 def Func(b: bool)
3679 Crash()
3680 enddef
3681
3682 def Crash()
3683 sil! :/not found/d _
3684 sil! :/not found/put _
3685 enddef
3686
3687 Func(true)
3688 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003689 v9.CheckScriptSuccess(lines)
Bram Moolenaarf9041332021-01-21 19:41:16 +01003690enddef
3691
Bram Moolenaar5b3d1bb2020-12-22 12:20:08 +01003692def Test_opfunc()
Bram Moolenaar848fadd2022-01-30 15:28:30 +00003693 nnoremap <F3> <cmd>set opfunc=g:Opfunc<cr>g@
Bram Moolenaar5b3d1bb2020-12-22 12:20:08 +01003694 def g:Opfunc(_: any): string
3695 setline(1, 'ASDF')
3696 return ''
3697 enddef
3698 new
3699 setline(1, 'asdf')
3700 feedkeys("\<F3>$", 'x')
3701 assert_equal('ASDF', getline(1))
3702
3703 bwipe!
3704 nunmap <F3>
3705enddef
3706
Bram Moolenaar3b309f12021-12-13 18:19:55 +00003707func Test_opfunc_error()
3708 CheckScreendump
3709 call Run_Test_opfunc_error()
3710endfunc
3711
3712def Run_Test_opfunc_error()
3713 # test that the error from Opfunc() is displayed right away
3714 var lines =<< trim END
3715 vim9script
3716
3717 def Opfunc(type: string)
3718 try
3719 eval [][0]
3720 catch /nothing/ # error not caught
3721 endtry
3722 enddef
3723 &operatorfunc = Opfunc
3724 nnoremap <expr> l <SID>L()
3725 def L(): string
3726 return 'l'
3727 enddef
3728 'x'->repeat(10)->setline(1)
3729 feedkeys('g@l', 'n')
3730 feedkeys('llll')
3731 END
3732 call writefile(lines, 'XTest_opfunc_error')
3733
Bram Moolenaar62aec932022-01-29 21:45:34 +00003734 var buf = g:RunVimInTerminal('-S XTest_opfunc_error', {rows: 6, wait_for_ruler: 0})
3735 g:WaitForAssert(() => assert_match('Press ENTER', term_getline(buf, 6)))
Bram Moolenaarec892232022-05-06 17:53:06 +01003736 g:WaitForAssert(() => assert_match('E684: List index out of range: 0', term_getline(buf, 5)))
Bram Moolenaar3b309f12021-12-13 18:19:55 +00003737
3738 # clean up
Bram Moolenaar62aec932022-01-29 21:45:34 +00003739 g:StopVimInTerminal(buf)
Bram Moolenaar3b309f12021-12-13 18:19:55 +00003740 delete('XTest_opfunc_error')
3741enddef
3742
Bram Moolenaar077a4232020-12-22 18:33:27 +01003743" this was crashing on exit
3744def Test_nested_lambda_in_closure()
3745 var lines =<< trim END
3746 vim9script
Bram Moolenaar227c58a2021-04-28 20:40:44 +02003747 command WriteDone writefile(['Done'], 'XnestedDone')
Bram Moolenaar077a4232020-12-22 18:33:27 +01003748 def Outer()
3749 def g:Inner()
3750 echo map([1, 2, 3], {_, v -> v + 1})
3751 enddef
3752 g:Inner()
3753 enddef
3754 defcompile
Bram Moolenaar227c58a2021-04-28 20:40:44 +02003755 # not reached
Bram Moolenaar077a4232020-12-22 18:33:27 +01003756 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003757 if !g:RunVim([], lines, '--clean -c WriteDone -c quit')
Bram Moolenaar077a4232020-12-22 18:33:27 +01003758 return
3759 endif
3760 assert_equal(['Done'], readfile('XnestedDone'))
3761 delete('XnestedDone')
3762enddef
3763
Bram Moolenaar92368aa2022-02-07 17:50:39 +00003764def Test_nested_closure_funcref()
3765 var lines =<< trim END
3766 vim9script
3767 def Func()
3768 var n: number
3769 def Nested()
3770 ++n
3771 enddef
3772 Nested()
3773 g:result_one = n
3774 var Ref = function(Nested)
3775 Ref()
3776 g:result_two = n
3777 enddef
3778 Func()
3779 END
3780 v9.CheckScriptSuccess(lines)
3781 assert_equal(1, g:result_one)
3782 assert_equal(2, g:result_two)
3783 unlet g:result_one g:result_two
3784enddef
3785
Bram Moolenaar7aca5ca2022-02-07 19:56:43 +00003786def Test_nested_closure_in_dict()
3787 var lines =<< trim END
3788 vim9script
3789 def Func(): dict<any>
3790 var n: number
3791 def Inc(): number
3792 ++n
3793 return n
3794 enddef
3795 return {inc: function(Inc)}
3796 enddef
3797 disas Func
3798 var d = Func()
3799 assert_equal(1, d.inc())
3800 assert_equal(2, d.inc())
3801 END
3802 v9.CheckScriptSuccess(lines)
3803enddef
3804
Bram Moolenaarfb43cfc2022-03-11 18:54:17 +00003805def Test_script_local_other_script()
3806 var lines =<< trim END
3807 function LegacyJob()
3808 let FuncRef = function('s:close_cb')
3809 endfunction
3810 function s:close_cb(...)
3811 endfunction
3812 END
3813 lines->writefile('Xlegacy.vim')
3814 source Xlegacy.vim
3815 g:LegacyJob()
3816 g:LegacyJob()
3817 g:LegacyJob()
3818
3819 delfunc g:LegacyJob
3820 delete('Xlegacy.vim')
3821enddef
3822
Bram Moolenaar04947cc2021-03-06 19:26:46 +01003823def Test_check_func_arg_types()
3824 var lines =<< trim END
3825 vim9script
3826 def F1(x: string): string
3827 return x
3828 enddef
3829
3830 def F2(x: number): number
3831 return x + 1
3832 enddef
3833
Bram Moolenaar1d9cef72022-03-17 16:30:03 +00003834 def G(Fg: func): dict<func>
3835 return {f: Fg}
Bram Moolenaar04947cc2021-03-06 19:26:46 +01003836 enddef
3837
3838 def H(d: dict<func>): string
3839 return d.f('a')
3840 enddef
3841 END
3842
Bram Moolenaar62aec932022-01-29 21:45:34 +00003843 v9.CheckScriptSuccess(lines + ['echo H(G(F1))'])
3844 v9.CheckScriptFailure(lines + ['echo H(G(F2))'], 'E1013:')
Bram Moolenaar1d9cef72022-03-17 16:30:03 +00003845
3846 v9.CheckScriptFailure(lines + ['def SomeFunc(ff: func)', 'enddef'], 'E704:')
Bram Moolenaar04947cc2021-03-06 19:26:46 +01003847enddef
3848
Bram Moolenaarbadf04f2022-03-12 21:28:22 +00003849def Test_call_func_with_null()
3850 var lines =<< trim END
3851 def Fstring(v: string)
3852 assert_equal(null_string, v)
3853 enddef
3854 Fstring(null_string)
3855 def Fblob(v: blob)
3856 assert_equal(null_blob, v)
3857 enddef
3858 Fblob(null_blob)
3859 def Flist(v: list<number>)
3860 assert_equal(null_list, v)
3861 enddef
3862 Flist(null_list)
3863 def Fdict(v: dict<number>)
3864 assert_equal(null_dict, v)
3865 enddef
3866 Fdict(null_dict)
3867 def Ffunc(Fv: func(number): number)
3868 assert_equal(null_function, Fv)
3869 enddef
3870 Ffunc(null_function)
3871 if has('channel')
3872 def Fchannel(v: channel)
3873 assert_equal(null_channel, v)
3874 enddef
3875 Fchannel(null_channel)
3876 def Fjob(v: job)
3877 assert_equal(null_job, v)
3878 enddef
3879 Fjob(null_job)
3880 endif
3881 END
3882 v9.CheckDefAndScriptSuccess(lines)
3883enddef
3884
3885def Test_null_default_argument()
3886 var lines =<< trim END
3887 def Fstring(v: string = null_string)
3888 assert_equal(null_string, v)
3889 enddef
3890 Fstring()
3891 def Fblob(v: blob = null_blob)
3892 assert_equal(null_blob, v)
3893 enddef
3894 Fblob()
3895 def Flist(v: list<number> = null_list)
3896 assert_equal(null_list, v)
3897 enddef
3898 Flist()
3899 def Fdict(v: dict<number> = null_dict)
3900 assert_equal(null_dict, v)
3901 enddef
3902 Fdict()
3903 def Ffunc(Fv: func(number): number = null_function)
3904 assert_equal(null_function, Fv)
3905 enddef
3906 Ffunc()
3907 if has('channel')
3908 def Fchannel(v: channel = null_channel)
3909 assert_equal(null_channel, v)
3910 enddef
3911 Fchannel()
3912 def Fjob(v: job = null_job)
3913 assert_equal(null_job, v)
3914 enddef
3915 Fjob()
3916 endif
3917 END
3918 v9.CheckDefAndScriptSuccess(lines)
3919enddef
3920
3921def Test_null_return()
3922 var lines =<< trim END
3923 def Fstring(): string
3924 return null_string
3925 enddef
3926 assert_equal(null_string, Fstring())
3927 def Fblob(): blob
3928 return null_blob
3929 enddef
3930 assert_equal(null_blob, Fblob())
3931 def Flist(): list<number>
3932 return null_list
3933 enddef
3934 assert_equal(null_list, Flist())
3935 def Fdict(): dict<number>
3936 return null_dict
3937 enddef
3938 assert_equal(null_dict, Fdict())
3939 def Ffunc(): func(number): number
3940 return null_function
3941 enddef
3942 assert_equal(null_function, Ffunc())
3943 if has('channel')
3944 def Fchannel(): channel
3945 return null_channel
3946 enddef
3947 assert_equal(null_channel, Fchannel())
3948 def Fjob(): job
3949 return null_job
3950 enddef
3951 assert_equal(null_job, Fjob())
3952 endif
3953 END
3954 v9.CheckDefAndScriptSuccess(lines)
3955enddef
3956
Bram Moolenaar6e48b842021-08-10 22:52:02 +02003957def Test_list_any_type_checked()
3958 var lines =<< trim END
3959 vim9script
3960 def Foo()
3961 --decl--
3962 Bar(l)
3963 enddef
3964 def Bar(ll: list<dict<any>>)
3965 enddef
3966 Foo()
3967 END
Bram Moolenaar2d3ac2e2022-02-03 12:34:05 +00003968 # "any" could be "dict<any>", thus OK
Bram Moolenaar6e48b842021-08-10 22:52:02 +02003969 lines[2] = 'var l: list<any>'
Bram Moolenaar2d3ac2e2022-02-03 12:34:05 +00003970 v9.CheckScriptSuccess(lines)
Bram Moolenaar6e48b842021-08-10 22:52:02 +02003971 lines[2] = 'var l: list<any> = []'
Bram Moolenaar2d3ac2e2022-02-03 12:34:05 +00003972 v9.CheckScriptSuccess(lines)
Bram Moolenaar6e48b842021-08-10 22:52:02 +02003973
3974 lines[2] = 'var l: list<any> = [11]'
Bram Moolenaar62aec932022-01-29 21:45:34 +00003975 v9.CheckScriptFailure(lines, 'E1013: Argument 1: type mismatch, expected list<dict<any>> but got list<number>', 2)
Bram Moolenaar6e48b842021-08-10 22:52:02 +02003976enddef
3977
Bram Moolenaar701cc6c2021-04-10 13:33:48 +02003978def Test_compile_error()
3979 var lines =<< trim END
3980 def g:Broken()
3981 echo 'a' + {}
3982 enddef
3983 call g:Broken()
3984 END
3985 # First call: compilation error
Bram Moolenaar62aec932022-01-29 21:45:34 +00003986 v9.CheckScriptFailure(lines, 'E1051: Wrong argument type for +')
Bram Moolenaar701cc6c2021-04-10 13:33:48 +02003987
3988 # Second call won't try compiling again
3989 assert_fails('call g:Broken()', 'E1091: Function is not compiled: Broken')
Bram Moolenaar599410c2021-04-10 14:03:43 +02003990 delfunc g:Broken
3991
3992 # No error when compiling with :silent!
3993 lines =<< trim END
3994 def g:Broken()
3995 echo 'a' + []
3996 enddef
3997 silent! defcompile
3998 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003999 v9.CheckScriptSuccess(lines)
Bram Moolenaar599410c2021-04-10 14:03:43 +02004000
4001 # Calling the function won't try compiling again
4002 assert_fails('call g:Broken()', 'E1091: Function is not compiled: Broken')
4003 delfunc g:Broken
Bram Moolenaar701cc6c2021-04-10 13:33:48 +02004004enddef
4005
Bram Moolenaar962c43b2021-04-10 17:18:09 +02004006def Test_ignored_argument()
4007 var lines =<< trim END
4008 vim9script
4009 def Ignore(_, _): string
4010 return 'yes'
4011 enddef
4012 assert_equal('yes', Ignore(1, 2))
4013
4014 func Ok(_)
4015 return a:_
4016 endfunc
4017 assert_equal('ok', Ok('ok'))
4018
4019 func Oktoo()
4020 let _ = 'too'
4021 return _
4022 endfunc
4023 assert_equal('too', Oktoo())
Bram Moolenaarda479c72021-04-10 21:01:38 +02004024
4025 assert_equal([[1], [2], [3]], range(3)->mapnew((_, v) => [v]->map((_, w) => w + 1)))
Bram Moolenaar962c43b2021-04-10 17:18:09 +02004026 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00004027 v9.CheckScriptSuccess(lines)
Bram Moolenaar962c43b2021-04-10 17:18:09 +02004028
4029 lines =<< trim END
4030 def Ignore(_: string): string
4031 return _
4032 enddef
4033 defcompile
4034 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00004035 v9.CheckScriptFailure(lines, 'E1181:', 1)
Bram Moolenaar962c43b2021-04-10 17:18:09 +02004036
4037 lines =<< trim END
4038 var _ = 1
4039 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00004040 v9.CheckDefAndScriptFailure(lines, 'E1181:', 1)
Yegappan Lakshmanan34fcb692021-05-25 20:14:00 +02004041
4042 lines =<< trim END
4043 var x = _
4044 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00004045 v9.CheckDefAndScriptFailure(lines, 'E1181:', 1)
Bram Moolenaar962c43b2021-04-10 17:18:09 +02004046enddef
4047
Bram Moolenaarbb8a7ce2021-04-10 20:10:26 +02004048def Test_too_many_arguments()
4049 var lines =<< trim END
4050 echo [0, 1, 2]->map(() => 123)
4051 END
Bram Moolenaareddd4fc2022-02-20 15:52:28 +00004052 v9.CheckDefAndScriptFailure(lines, ['E176:', 'E1106: 2 arguments too many'], 1)
Bram Moolenaarbb8a7ce2021-04-10 20:10:26 +02004053
4054 lines =<< trim END
4055 echo [0, 1, 2]->map((_) => 123)
4056 END
Bram Moolenaareddd4fc2022-02-20 15:52:28 +00004057 v9.CheckDefAndScriptFailure(lines, ['E176', 'E1106: One argument too many'], 1)
Bram Moolenaar31d99482022-05-26 22:24:43 +01004058
4059 lines =<< trim END
4060 vim9script
4061 def OneArgument(arg: string)
4062 echo arg
4063 enddef
4064 var Ref = OneArgument
4065 Ref('a', 'b')
4066 END
4067 v9.CheckScriptFailure(lines, 'E118:')
4068enddef
4069
4070def Test_funcref_with_base()
4071 var lines =<< trim END
4072 vim9script
4073 def TwoArguments(str: string, nr: number)
4074 echo str nr
4075 enddef
4076 var Ref = TwoArguments
4077 Ref('a', 12)
4078 'b'->Ref(34)
4079 END
4080 v9.CheckScriptSuccess(lines)
4081
4082 lines =<< trim END
4083 vim9script
4084 def TwoArguments(str: string, nr: number)
4085 echo str nr
4086 enddef
4087 var Ref = TwoArguments
4088 'a'->Ref('b')
4089 END
4090 v9.CheckScriptFailure(lines, 'E1013: Argument 2: type mismatch, expected number but got string', 6)
4091
4092 lines =<< trim END
4093 vim9script
4094 def TwoArguments(str: string, nr: number)
4095 echo str nr
4096 enddef
4097 var Ref = TwoArguments
4098 123->Ref(456)
4099 END
4100 v9.CheckScriptFailure(lines, 'E1013: Argument 1: type mismatch, expected string but got number')
4101
4102 lines =<< trim END
4103 vim9script
4104 def TwoArguments(nr: number, str: string)
4105 echo str nr
4106 enddef
4107 var Ref = TwoArguments
4108 123->Ref('b')
4109 def AndNowCompiled()
4110 456->Ref('x')
4111 enddef
4112 AndNowCompiled()
4113 END
4114 v9.CheckScriptSuccess(lines)
Bram Moolenaarbb8a7ce2021-04-10 20:10:26 +02004115enddef
Bram Moolenaar077a4232020-12-22 18:33:27 +01004116
Bram Moolenaara6aa1642021-04-23 19:32:23 +02004117def Test_closing_brace_at_start_of_line()
4118 var lines =<< trim END
4119 def Func()
4120 enddef
4121 Func(
4122 )
4123 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00004124 v9.CheckDefAndScriptSuccess(lines)
Bram Moolenaara6aa1642021-04-23 19:32:23 +02004125enddef
4126
Bram Moolenaar62aec932022-01-29 21:45:34 +00004127func s:CreateMydict()
Bram Moolenaarb033ee22021-08-15 16:08:36 +02004128 let g:mydict = {}
4129 func g:mydict.afunc()
4130 let g:result = self.key
4131 endfunc
4132endfunc
4133
4134def Test_numbered_function_reference()
4135 CreateMydict()
4136 var output = execute('legacy func g:mydict.afunc')
4137 var funcName = 'g:' .. substitute(output, '.*function \(\d\+\).*', '\1', '')
4138 execute 'function(' .. funcName .. ', [], {key: 42})()'
4139 # check that the function still exists
4140 assert_equal(output, execute('legacy func g:mydict.afunc'))
4141 unlet g:mydict
4142enddef
4143
Bram Moolenaard3a11782022-01-05 16:50:40 +00004144def Test_go_beyond_end_of_cmd()
4145 # this was reading the byte after the end of the line
4146 var lines =<< trim END
4147 def F()
4148 cal
4149 enddef
4150 defcompile
4151 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00004152 v9.CheckScriptFailure(lines, 'E476:')
Bram Moolenaard3a11782022-01-05 16:50:40 +00004153enddef
4154
Yegappan Lakshmanan7c7e19c2022-04-09 11:09:07 +01004155" Test for memory allocation failure when defining a new lambda
4156func Test_lambda_allocation_failure()
4157 new
4158 let lines =<< trim END
4159 vim9script
4160 g:Xlambda = (x): number => {
4161 return x + 1
4162 }
4163 END
4164 call setline(1, lines)
4165 call test_alloc_fail(GetAllocId('get_func'), 0, 0)
4166 call assert_fails('source', 'E342:')
4167 call assert_false(exists('g:Xlambda'))
4168 bw!
4169endfunc
4170
Bram Moolenaarbce69d62022-05-22 13:45:52 +01004171def Test_multiple_funcref()
4172 # This was using a NULL pointer
4173 var lines =<< trim END
4174 vim9script
4175 def A(F: func, ...args: list<any>): func
4176 return funcref(F, args)
4177 enddef
4178
4179 def B(F: func): func
4180 return funcref(A, [F])
4181 enddef
4182
4183 def Test(n: number)
4184 enddef
4185
4186 const X = B(Test)
4187 X(1)
4188 END
4189 v9.CheckScriptSuccess(lines)
4190
4191 # slightly different case
4192 lines =<< trim END
4193 vim9script
4194
4195 def A(F: func, ...args: list<any>): any
4196 return call(F, args)
4197 enddef
4198
4199 def B(F: func): func
4200 return funcref(A, [F])
4201 enddef
4202
4203 def Test(n: number)
4204 enddef
4205
4206 const X = B(Test)
4207 X(1)
4208 END
4209 v9.CheckScriptSuccess(lines)
4210enddef
4211
Bram Moolenaarbd683e32022-07-18 17:49:03 +01004212def Test_cexpr_errmsg_line_number()
4213 var lines =<< trim END
4214 vim9script
4215 def Func()
4216 var qfl = {}
4217 cexpr qfl
4218 enddef
4219 Func()
4220 END
4221 v9.CheckScriptFailure(lines, 'E777', 2)
4222enddef
4223
Bram Moolenaar8b716f52022-02-15 21:17:56 +00004224" The following messes up syntax highlight, keep near the end.
Bram Moolenaar20677332021-06-06 17:02:53 +02004225if has('python3')
Bram Moolenaar8b716f52022-02-15 21:17:56 +00004226 def Test_python3_command()
4227 py3 import vim
Bram Moolenaarf5288c52022-02-15 21:33:29 +00004228 py3 vim.command("g:done = 'yes'")
Bram Moolenaar8b716f52022-02-15 21:17:56 +00004229 assert_equal('yes', g:done)
4230 unlet g:done
4231 enddef
4232
Bram Moolenaar20677332021-06-06 17:02:53 +02004233 def Test_python3_heredoc()
4234 py3 << trim EOF
4235 import vim
4236 vim.vars['didit'] = 'yes'
4237 EOF
4238 assert_equal('yes', g:didit)
4239
4240 python3 << trim EOF
4241 import vim
4242 vim.vars['didit'] = 'again'
4243 EOF
4244 assert_equal('again', g:didit)
4245 enddef
4246endif
4247
Bram Moolenaar20677332021-06-06 17:02:53 +02004248if has('lua')
4249 def Test_lua_heredoc()
4250 g:d = {}
4251 lua << trim EOF
4252 x = vim.eval('g:d')
4253 x['key'] = 'val'
4254 EOF
4255 assert_equal('val', g:d.key)
4256 enddef
Bram Moolenaarefd73ae2022-03-20 18:51:00 +00004257
4258 def Test_lua_heredoc_fails()
4259 var lines = [
4260 'vim9script',
4261 'def ExeLua()',
4262 'lua << trim EOLUA',
4263 "x = vim.eval('g:nodict')",
4264 'EOLUA',
4265 'enddef',
4266 'ExeLua()',
4267 ]
4268 v9.CheckScriptFailure(lines, 'E121: Undefined variable: g:nodict')
4269 enddef
Bram Moolenaar20677332021-06-06 17:02:53 +02004270endif
4271
Bram Moolenaard881d152022-05-13 13:50:36 +01004272if has('perl')
4273 def Test_perl_heredoc_nested()
4274 var lines =<< trim END
4275 vim9script
4276 def F(): string
4277 def G(): string
4278 perl << EOF
4279 EOF
4280 return 'done'
4281 enddef
4282 return G()
4283 enddef
4284 assert_equal('done', F())
4285 END
4286 v9.CheckScriptSuccess(lines)
4287 enddef
4288endif
4289
Bram Moolenaarf7779c62020-05-03 15:38:16 +02004290
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02004291" vim: ts=8 sw=2 sts=2 expandtab tw=80 fdm=marker