blob: 14d0bce6f510d2d5fd3bc64fc737cedae1cdf861 [file] [log] [blame]
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001" Test various aspects of the Vim9 script language.
2
3source check.vim
Bram Moolenaarad304702020-09-06 18:22:53 +02004source term_util.vim
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02005source view_util.vim
Bram Moolenaar62aec932022-01-29 21:45:34 +00006import './vim9.vim' as v9
Bram Moolenaar47e7d702020-07-05 18:18:42 +02007source screendump.vim
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02008
9func Test_def_basic()
10 def SomeFunc(): string
11 return 'yes'
12 enddef
Bram Moolenaarc0c71e92020-09-11 19:09:48 +020013 call SomeFunc()->assert_equal('yes')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +020014endfunc
15
Bram Moolenaar2b9b17e2020-10-13 18:38:11 +020016func Test_compiling_error()
17 " use a terminal to see the whole error message
Bram Moolenaarf4e8cdd2020-10-12 22:07:13 +020018 CheckRunVimInTerminal
19
Bram Moolenaar2b9b17e2020-10-13 18:38:11 +020020 call TestCompilingError()
Bram Moolenaare8c46602021-04-05 22:27:37 +020021 call TestCompilingErrorInTry()
Bram Moolenaar2b9b17e2020-10-13 18:38:11 +020022endfunc
23
24def TestCompilingError()
Bram Moolenaarf4e8cdd2020-10-12 22:07:13 +020025 var lines =<< trim END
26 vim9script
27 def Fails()
28 echo nothing
29 enddef
30 defcompile
31 END
Bram Moolenaare8c46602021-04-05 22:27:37 +020032 writefile(lines, 'XTest_compile_error')
Bram Moolenaar62aec932022-01-29 21:45:34 +000033 var buf = g:RunVimInTerminal('-S XTest_compile_error',
Bram Moolenaare0de1712020-12-02 17:36:54 +010034 {rows: 10, wait_for_ruler: 0})
Bram Moolenaar62aec932022-01-29 21:45:34 +000035 g:WaitForAssert(() => assert_match('Error detected while compiling command line.*Fails.*Variable not found: nothing',
36 g:Term_getlines(buf, range(1, 9))))
Bram Moolenaarf4e8cdd2020-10-12 22:07:13 +020037
38 # clean up
Bram Moolenaar62aec932022-01-29 21:45:34 +000039 g:StopVimInTerminal(buf)
Bram Moolenaare8c46602021-04-05 22:27:37 +020040 delete('XTest_compile_error')
41enddef
42
43def TestCompilingErrorInTry()
44 var dir = 'Xdir/autoload'
45 mkdir(dir, 'p')
46
47 var lines =<< trim END
48 vim9script
Bram Moolenaard8fe6d32022-01-30 18:40:44 +000049 export def OnlyCompiled()
Bram Moolenaare8c46602021-04-05 22:27:37 +020050 g:runtime = 'yes'
51 invalid
52 enddef
53 END
54 writefile(lines, dir .. '/script.vim')
55
56 lines =<< trim END
57 vim9script
58 todo
59 try
60 script#OnlyCompiled()
61 catch /nothing/
62 endtry
63 END
64 lines[1] = 'set rtp=' .. getcwd() .. '/Xdir'
65 writefile(lines, 'XTest_compile_error')
66
Bram Moolenaar62aec932022-01-29 21:45:34 +000067 var buf = g:RunVimInTerminal('-S XTest_compile_error', {rows: 10, wait_for_ruler: 0})
68 g:WaitForAssert(() => assert_match('Error detected while compiling command line.*function script#OnlyCompiled.*Invalid command: invalid',
69 g:Term_getlines(buf, range(1, 9))))
Bram Moolenaare8c46602021-04-05 22:27:37 +020070
71 # clean up
Bram Moolenaar62aec932022-01-29 21:45:34 +000072 g:StopVimInTerminal(buf)
Bram Moolenaare8c46602021-04-05 22:27:37 +020073 delete('XTest_compile_error')
74 delete('Xdir', 'rf')
Bram Moolenaarf4e8cdd2020-10-12 22:07:13 +020075enddef
76
Bram Moolenaarb55d6182021-06-08 22:01:53 +020077def Test_compile_error_in_called_function()
78 var lines =<< trim END
79 vim9script
80 var n: number
81 def Foo()
82 &hls = n
83 enddef
84 def Bar()
85 Foo()
86 enddef
87 silent! Foo()
88 Bar()
89 END
Bram Moolenaar62aec932022-01-29 21:45:34 +000090 v9.CheckScriptFailureList(lines, ['E1012:', 'E1191:'])
Bram Moolenaarb55d6182021-06-08 22:01:53 +020091enddef
92
Bram Moolenaar22f17a22021-06-21 20:48:58 +020093def Test_wrong_function_name()
94 var lines =<< trim END
95 vim9script
96 func _Foo()
97 echo 'foo'
98 endfunc
99 END
Bram Moolenaar3787f262022-02-07 21:54:01 +0000100 v9.CheckScriptFailure(lines, 'E1267:')
Bram Moolenaar22f17a22021-06-21 20:48:58 +0200101
102 lines =<< trim END
103 vim9script
104 def _Foo()
105 echo 'foo'
106 enddef
107 END
Bram Moolenaar3787f262022-02-07 21:54:01 +0000108 v9.CheckScriptFailure(lines, 'E1267:')
Bram Moolenaardea5ab02022-02-23 22:12:02 +0000109
110 lines =<< trim END
111 vim9script
112 var Object = {}
113 function Object.Method()
114 endfunction
115 END
116 v9.CheckScriptFailure(lines, 'E1182:')
117
118 lines =<< trim END
119 vim9script
120 var Object = {}
121 def Object.Method()
122 enddef
123 END
124 v9.CheckScriptFailure(lines, 'E1182:')
125
126 lines =<< trim END
127 vim9script
128 g:Object = {}
129 function g:Object.Method()
130 endfunction
131 END
132 v9.CheckScriptFailure(lines, 'E1182:')
133
134 lines =<< trim END
135 let s:Object = {}
136 def Define()
137 function s:Object.Method()
138 endfunction
139 enddef
140 defcompile
141 END
142 v9.CheckScriptFailure(lines, 'E1182:')
143 delfunc g:Define
144
145 lines =<< trim END
146 let s:Object = {}
147 def Define()
148 def Object.Method()
149 enddef
150 enddef
151 defcompile
152 END
153 v9.CheckScriptFailure(lines, 'E1182:')
154 delfunc g:Define
155
156 lines =<< trim END
157 let g:Object = {}
158 def Define()
159 function g:Object.Method()
160 endfunction
161 enddef
162 defcompile
163 END
164 v9.CheckScriptFailure(lines, 'E1182:')
165 delfunc g:Define
Bram Moolenaar22f17a22021-06-21 20:48:58 +0200166enddef
167
Bram Moolenaarf48b2fa2021-04-12 22:02:36 +0200168def Test_autoload_name_mismatch()
169 var dir = 'Xdir/autoload'
170 mkdir(dir, 'p')
171
172 var lines =<< trim END
173 vim9script
Bram Moolenaard8fe6d32022-01-30 18:40:44 +0000174 export def NoFunction()
Bram Moolenaarf48b2fa2021-04-12 22:02:36 +0200175 # comment
176 g:runtime = 'yes'
177 enddef
178 END
179 writefile(lines, dir .. '/script.vim')
180
181 var save_rtp = &rtp
182 exe 'set rtp=' .. getcwd() .. '/Xdir'
183 lines =<< trim END
184 call script#Function()
185 END
Bram Moolenaard8fe6d32022-01-30 18:40:44 +0000186 v9.CheckScriptFailure(lines, 'E117:', 1)
Bram Moolenaarf48b2fa2021-04-12 22:02:36 +0200187
188 &rtp = save_rtp
189 delete(dir, 'rf')
190enddef
191
Bram Moolenaarf0a40692021-06-11 22:05:47 +0200192def Test_autoload_names()
193 var dir = 'Xdir/autoload'
194 mkdir(dir, 'p')
195
196 var lines =<< trim END
197 func foobar#function()
198 return 'yes'
199 endfunc
200 let foobar#var = 'no'
201 END
202 writefile(lines, dir .. '/foobar.vim')
203
204 var save_rtp = &rtp
205 exe 'set rtp=' .. getcwd() .. '/Xdir'
206
207 lines =<< trim END
208 assert_equal('yes', foobar#function())
209 var Function = foobar#function
210 assert_equal('yes', Function())
211
212 assert_equal('no', foobar#var)
213 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000214 v9.CheckDefAndScriptSuccess(lines)
Bram Moolenaarf0a40692021-06-11 22:05:47 +0200215
216 &rtp = save_rtp
217 delete(dir, 'rf')
218enddef
219
Bram Moolenaar88c89c72021-08-14 14:01:05 +0200220def Test_autoload_error_in_script()
221 var dir = 'Xdir/autoload'
222 mkdir(dir, 'p')
223
224 var lines =<< trim END
225 func scripterror#function()
226 let g:called_function = 'yes'
227 endfunc
228 let 0 = 1
229 END
230 writefile(lines, dir .. '/scripterror.vim')
231
232 var save_rtp = &rtp
233 exe 'set rtp=' .. getcwd() .. '/Xdir'
234
235 g:called_function = 'no'
236 # The error in the autoload script cannot be checked with assert_fails(), use
237 # CheckDefSuccess() instead of CheckDefFailure()
238 try
Bram Moolenaar62aec932022-01-29 21:45:34 +0000239 v9.CheckDefSuccess(['scripterror#function()'])
Bram Moolenaar88c89c72021-08-14 14:01:05 +0200240 catch
241 assert_match('E121: Undefined variable: 0', v:exception)
242 endtry
243 assert_equal('no', g:called_function)
244
245 lines =<< trim END
246 func scriptcaught#function()
247 let g:called_function = 'yes'
248 endfunc
249 try
250 let 0 = 1
251 catch
252 let g:caught = v:exception
253 endtry
254 END
255 writefile(lines, dir .. '/scriptcaught.vim')
256
257 g:called_function = 'no'
Bram Moolenaar62aec932022-01-29 21:45:34 +0000258 v9.CheckDefSuccess(['scriptcaught#function()'])
Bram Moolenaar88c89c72021-08-14 14:01:05 +0200259 assert_match('E121: Undefined variable: 0', g:caught)
260 assert_equal('yes', g:called_function)
261
262 &rtp = save_rtp
263 delete(dir, 'rf')
264enddef
265
Bram Moolenaar62aec932022-01-29 21:45:34 +0000266def s:CallRecursive(n: number): number
Bram Moolenaar0ba48e82020-11-17 18:23:19 +0100267 return CallRecursive(n + 1)
268enddef
269
Bram Moolenaar62aec932022-01-29 21:45:34 +0000270def s:CallMapRecursive(l: list<number>): number
Bram Moolenaar2949cfd2020-12-31 21:28:47 +0100271 return map(l, (_, v) => CallMapRecursive([v]))[0]
Bram Moolenaar0ba48e82020-11-17 18:23:19 +0100272enddef
273
274def Test_funcdepth_error()
275 set maxfuncdepth=10
276
277 var caught = false
278 try
279 CallRecursive(1)
280 catch /E132:/
281 caught = true
282 endtry
283 assert_true(caught)
284
285 caught = false
286 try
287 CallMapRecursive([1])
288 catch /E132:/
289 caught = true
290 endtry
291 assert_true(caught)
292
293 set maxfuncdepth&
294enddef
295
Bram Moolenaar5178b1b2021-01-01 18:43:51 +0100296def Test_endfunc_enddef()
297 var lines =<< trim END
298 def Test()
299 echo 'test'
300 endfunc
301 enddef
302 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000303 v9.CheckScriptFailure(lines, 'E1151:', 3)
Bram Moolenaar5178b1b2021-01-01 18:43:51 +0100304
305 lines =<< trim END
306 def Test()
307 func Nested()
308 echo 'test'
309 enddef
310 enddef
311 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000312 v9.CheckScriptFailure(lines, 'E1152:', 4)
Bram Moolenaar49f1e9e2021-03-22 20:49:02 +0100313
314 lines =<< trim END
315 def Ok()
316 echo 'hello'
317 enddef | echo 'there'
318 def Bad()
319 echo 'hello'
320 enddef there
321 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000322 v9.CheckScriptFailure(lines, 'E1173: Text found after enddef: there', 6)
Bram Moolenaar5178b1b2021-01-01 18:43:51 +0100323enddef
324
Bram Moolenaarb8ba9b92021-01-01 18:54:34 +0100325def Test_missing_endfunc_enddef()
326 var lines =<< trim END
327 vim9script
328 def Test()
329 echo 'test'
330 endef
331 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000332 v9.CheckScriptFailure(lines, 'E1057:', 2)
Bram Moolenaarb8ba9b92021-01-01 18:54:34 +0100333
334 lines =<< trim END
335 vim9script
336 func Some()
337 echo 'test'
338 enfffunc
339 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000340 v9.CheckScriptFailure(lines, 'E126:', 2)
Bram Moolenaarb8ba9b92021-01-01 18:54:34 +0100341enddef
342
Bram Moolenaar4efd9942021-01-24 21:14:20 +0100343def Test_white_space_before_paren()
344 var lines =<< trim END
345 vim9script
346 def Test ()
347 echo 'test'
348 enddef
349 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000350 v9.CheckScriptFailure(lines, 'E1068:', 2)
Bram Moolenaar4efd9942021-01-24 21:14:20 +0100351
352 lines =<< trim END
353 vim9script
354 func Test ()
355 echo 'test'
356 endfunc
357 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000358 v9.CheckScriptFailure(lines, 'E1068:', 2)
Bram Moolenaar4efd9942021-01-24 21:14:20 +0100359
360 lines =<< trim END
361 def Test ()
362 echo 'test'
363 enddef
364 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000365 v9.CheckScriptFailure(lines, 'E1068:', 1)
Bram Moolenaar4efd9942021-01-24 21:14:20 +0100366
367 lines =<< trim END
368 func Test ()
369 echo 'test'
370 endfunc
371 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000372 v9.CheckScriptSuccess(lines)
Bram Moolenaar4efd9942021-01-24 21:14:20 +0100373enddef
374
Bram Moolenaar832ea892021-01-08 21:55:26 +0100375def Test_enddef_dict_key()
376 var d = {
377 enddef: 'x',
378 endfunc: 'y',
379 }
380 assert_equal({enddef: 'x', endfunc: 'y'}, d)
381enddef
382
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200383def ReturnString(): string
384 return 'string'
385enddef
386
387def ReturnNumber(): number
388 return 123
389enddef
390
391let g:notNumber = 'string'
392
393def ReturnGlobal(): number
394 return g:notNumber
395enddef
396
397def Test_return_something()
Bram Moolenaar62aec932022-01-29 21:45:34 +0000398 g:ReturnString()->assert_equal('string')
399 g:ReturnNumber()->assert_equal(123)
Bram Moolenaar848fadd2022-01-30 15:28:30 +0000400 assert_fails('g:ReturnGlobal()', 'E1012: Type mismatch; expected number but got string', '', 1, 'ReturnGlobal')
Bram Moolenaaref7aadb2022-01-18 18:46:07 +0000401
402 var lines =<< trim END
403 vim9script
404
405 def Msg()
406 echomsg 'in Msg()...'
407 enddef
408
409 def Func()
410 return Msg()
411 enddef
412 defcompile
413 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000414 v9.CheckScriptFailure(lines, 'E1096:')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200415enddef
416
Bram Moolenaare32e5162021-01-21 20:21:29 +0100417def Test_check_argument_type()
418 var lines =<< trim END
419 vim9script
420 def Val(a: number, b: number): number
421 return 0
422 enddef
423 def Func()
424 var x: any = true
425 Val(0, x)
426 enddef
427 disass Func
428 Func()
429 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000430 v9.CheckScriptFailure(lines, 'E1013: Argument 2: type mismatch, expected number but got bool', 2)
Bram Moolenaare32e5162021-01-21 20:21:29 +0100431enddef
432
Bram Moolenaarefd88552020-06-18 20:50:10 +0200433def Test_missing_return()
Bram Moolenaar62aec932022-01-29 21:45:34 +0000434 v9.CheckDefFailure(['def Missing(): number',
Bram Moolenaarefd88552020-06-18 20:50:10 +0200435 ' if g:cond',
436 ' echo "no return"',
437 ' else',
438 ' return 0',
439 ' endif'
440 'enddef'], 'E1027:')
Bram Moolenaar62aec932022-01-29 21:45:34 +0000441 v9.CheckDefFailure(['def Missing(): number',
Bram Moolenaarefd88552020-06-18 20:50:10 +0200442 ' if g:cond',
443 ' return 1',
444 ' else',
445 ' echo "no return"',
446 ' endif'
447 'enddef'], 'E1027:')
Bram Moolenaar62aec932022-01-29 21:45:34 +0000448 v9.CheckDefFailure(['def Missing(): number',
Bram Moolenaarefd88552020-06-18 20:50:10 +0200449 ' if g:cond',
450 ' return 1',
451 ' else',
452 ' return 2',
453 ' endif'
454 ' return 3'
455 'enddef'], 'E1095:')
456enddef
457
Bram Moolenaar403dc312020-10-17 19:29:51 +0200458def Test_return_bool()
459 var lines =<< trim END
460 vim9script
461 def MenuFilter(id: number, key: string): bool
462 return popup_filter_menu(id, key)
463 enddef
464 def YesnoFilter(id: number, key: string): bool
465 return popup_filter_yesno(id, key)
466 enddef
467 defcompile
468 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000469 v9.CheckScriptSuccess(lines)
Bram Moolenaar403dc312020-10-17 19:29:51 +0200470enddef
471
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200472let s:nothing = 0
473def ReturnNothing()
474 s:nothing = 1
475 if true
476 return
477 endif
478 s:nothing = 2
479enddef
480
481def Test_return_nothing()
Bram Moolenaar62aec932022-01-29 21:45:34 +0000482 g:ReturnNothing()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +0200483 s:nothing->assert_equal(1)
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200484enddef
485
Bram Moolenaar648ea762021-01-15 19:04:32 +0100486def Test_return_invalid()
487 var lines =<< trim END
488 vim9script
489 def Func(): invalid
490 return xxx
491 enddef
492 defcompile
493 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000494 v9.CheckScriptFailure(lines, 'E1010:', 2)
Bram Moolenaar31842cd2021-02-12 22:10:21 +0100495
496 lines =<< trim END
497 vim9script
498 def Test(Fun: func(number): number): list<number>
499 return map([1, 2, 3], (_, i) => Fun(i))
500 enddef
501 defcompile
502 def Inc(nr: number): nr
503 return nr + 2
504 enddef
505 echo Test(Inc)
506 END
507 # doing this twice was leaking memory
Bram Moolenaar62aec932022-01-29 21:45:34 +0000508 v9.CheckScriptFailure(lines, 'E1010:')
509 v9.CheckScriptFailure(lines, 'E1010:')
Bram Moolenaar648ea762021-01-15 19:04:32 +0100510enddef
511
Bram Moolenaarefc084e2021-09-09 22:30:52 +0200512def Test_return_list_any()
Bram Moolenaar114dbda2022-01-03 12:28:03 +0000513 # This used to fail but now the actual list type is checked, and since it has
514 # an item of type string it can be used as list<string>.
Bram Moolenaarefc084e2021-09-09 22:30:52 +0200515 var lines =<< trim END
516 vim9script
517 def Func(): list<string>
518 var l: list<any>
519 l->add('string')
520 return l
521 enddef
522 echo Func()
523 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000524 v9.CheckScriptSuccess(lines)
Bram Moolenaar114dbda2022-01-03 12:28:03 +0000525
Bram Moolenaarefc084e2021-09-09 22:30:52 +0200526 lines =<< trim END
527 vim9script
528 def Func(): list<string>
529 var l: list<any>
530 l += ['string']
531 return l
532 enddef
533 echo Func()
534 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000535 v9.CheckScriptSuccess(lines)
Bram Moolenaarefc084e2021-09-09 22:30:52 +0200536enddef
537
Bram Moolenaar1a572e92022-03-15 12:28:10 +0000538def Test_return_any_two_types()
539 var lines =<< trim END
540 vim9script
541
542 def G(Fn: func(string): any)
543 g:result = Fn("hello")
544 enddef
545
546 def F(a: number, b: string): any
547 echo b
548 if a > 0
549 return 1
550 else
551 return []
552 endif
553 enddef
554
555 G(function(F, [1]))
556 END
557 v9.CheckScriptSuccess(lines)
558 assert_equal(1, g:result)
559 unlet g:result
560enddef
561
Bram Moolenaar62aec932022-01-29 21:45:34 +0000562func s:Increment()
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200563 let g:counter += 1
564endfunc
565
566def Test_call_ufunc_count()
567 g:counter = 1
568 Increment()
569 Increment()
570 Increment()
Bram Moolenaarf5be8cd2020-07-17 20:36:00 +0200571 # works with and without :call
Bram Moolenaarc0c71e92020-09-11 19:09:48 +0200572 g:counter->assert_equal(4)
573 eval g:counter->assert_equal(4)
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200574 unlet g:counter
575enddef
576
Bram Moolenaar1983f1a2022-02-28 20:55:02 +0000577def Test_call_ufunc_failure()
578 var lines =<< trim END
579 vim9script
580 def Tryit()
581 g:Global(1, 2, 3)
582 enddef
583
584 func g:Global(a, b, c)
585 echo a:a a:b a:c
586 endfunc
587
588 defcompile
589
590 func! g:Global(a, b)
591 echo a:a a:b
592 endfunc
593 Tryit()
594 END
595 v9.CheckScriptFailure(lines, 'E118: Too many arguments for function: Global')
596 delfunc g:Global
597
598 lines =<< trim END
599 vim9script
600
601 g:Ref = function('len')
602 def Tryit()
603 g:Ref('x')
604 enddef
605
606 defcompile
607
608 g:Ref = function('add')
609 Tryit()
610 END
611 v9.CheckScriptFailure(lines, 'E119: Not enough arguments for function: add')
612 unlet g:Ref
613enddef
614
Bram Moolenaar62aec932022-01-29 21:45:34 +0000615def s:MyVarargs(arg: string, ...rest: list<string>): string
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +0200616 var res = arg
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200617 for s in rest
618 res ..= ',' .. s
619 endfor
620 return res
621enddef
622
623def Test_call_varargs()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +0200624 MyVarargs('one')->assert_equal('one')
625 MyVarargs('one', 'two')->assert_equal('one,two')
626 MyVarargs('one', 'two', 'three')->assert_equal('one,two,three')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200627enddef
628
Bram Moolenaar01dd6c32021-09-05 16:36:23 +0200629def Test_call_white_space()
Bram Moolenaar62aec932022-01-29 21:45:34 +0000630 v9.CheckDefAndScriptFailure(["call Test ('text')"], ['E476:', 'E1068:'])
Bram Moolenaar01dd6c32021-09-05 16:36:23 +0200631enddef
632
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200633def MyDefaultArgs(name = 'string'): string
634 return name
635enddef
636
Bram Moolenaar62aec932022-01-29 21:45:34 +0000637def s:MyDefaultSecond(name: string, second: bool = true): string
Bram Moolenaare30f64b2020-07-15 19:48:20 +0200638 return second ? name : 'none'
639enddef
640
Bram Moolenaar38a3bfa2021-03-29 22:14:55 +0200641
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200642def Test_call_default_args()
Bram Moolenaar62aec932022-01-29 21:45:34 +0000643 g:MyDefaultArgs()->assert_equal('string')
644 g:MyDefaultArgs(v:none)->assert_equal('string')
645 g:MyDefaultArgs('one')->assert_equal('one')
646 assert_fails('g:MyDefaultArgs("one", "two")', 'E118:', '', 4, 'Test_call_default_args')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200647
Bram Moolenaarc0c71e92020-09-11 19:09:48 +0200648 MyDefaultSecond('test')->assert_equal('test')
649 MyDefaultSecond('test', true)->assert_equal('test')
650 MyDefaultSecond('test', false)->assert_equal('none')
Bram Moolenaare30f64b2020-07-15 19:48:20 +0200651
Bram Moolenaar38a3bfa2021-03-29 22:14:55 +0200652 var lines =<< trim END
653 def MyDefaultThird(name: string, aa = 'aa', bb = 'bb'): string
654 return name .. aa .. bb
655 enddef
656
657 MyDefaultThird('->')->assert_equal('->aabb')
658 MyDefaultThird('->', v:none)->assert_equal('->aabb')
659 MyDefaultThird('->', 'xx')->assert_equal('->xxbb')
660 MyDefaultThird('->', v:none, v:none)->assert_equal('->aabb')
661 MyDefaultThird('->', 'xx', v:none)->assert_equal('->xxbb')
662 MyDefaultThird('->', v:none, 'yy')->assert_equal('->aayy')
663 MyDefaultThird('->', 'xx', 'yy')->assert_equal('->xxyy')
Bram Moolenaare28d9b32021-07-03 18:56:53 +0200664
665 def DefArg(mandatory: any, optional = mandatory): string
666 return mandatory .. optional
667 enddef
668 DefArg(1234)->assert_equal('12341234')
669 DefArg("ok")->assert_equal('okok')
Bram Moolenaar38a3bfa2021-03-29 22:14:55 +0200670 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000671 v9.CheckDefAndScriptSuccess(lines)
Bram Moolenaar38a3bfa2021-03-29 22:14:55 +0200672
Bram Moolenaar62aec932022-01-29 21:45:34 +0000673 v9.CheckScriptFailure(['def Func(arg: number = asdf)', 'enddef', 'defcompile'], 'E1001:')
Bram Moolenaar2d870f82020-12-05 13:41:01 +0100674 delfunc g:Func
Bram Moolenaar62aec932022-01-29 21:45:34 +0000675 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 +0100676 delfunc g:Func
Bram Moolenaar62aec932022-01-29 21:45:34 +0000677 v9.CheckDefFailure(['def Func(x: number = )', 'enddef'], 'E15:')
Bram Moolenaar12bce952021-03-11 20:04:04 +0100678
Bram Moolenaar38a3bfa2021-03-29 22:14:55 +0200679 lines =<< trim END
Bram Moolenaar12bce952021-03-11 20:04:04 +0100680 vim9script
681 def Func(a = b == 0 ? 1 : 2, b = 0)
682 enddef
683 defcompile
684 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000685 v9.CheckScriptFailure(lines, 'E1001: Variable not found: b')
Bram Moolenaar59618fe2021-12-21 12:32:17 +0000686
Bram Moolenaarfa46ead2021-12-22 13:18:39 +0000687 # using script variable requires matching type or type cast when executed
Bram Moolenaar59618fe2021-12-21 12:32:17 +0000688 lines =<< trim END
689 vim9script
690 var a: any
691 def Func(arg: string = a)
692 echo arg
693 enddef
694 defcompile
695 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000696 v9.CheckScriptSuccess(lines + ['a = "text"', 'Func()'])
697 v9.CheckScriptFailure(lines + ['a = 123', 'Func()'], 'E1013: Argument 1: type mismatch, expected string but got number')
Bram Moolenaar59618fe2021-12-21 12:32:17 +0000698
699 # using global variable does not require type cast
700 lines =<< trim END
701 vim9script
702 def Func(arg: string = g:str)
703 echo arg
704 enddef
705 g:str = 'works'
706 Func()
707 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000708 v9.CheckScriptSuccess(lines)
Bram Moolenaar04b12692020-05-04 23:24:44 +0200709enddef
710
Bram Moolenaar62aec932022-01-29 21:45:34 +0000711def s:FuncWithComment( # comment
Bram Moolenaarcef12702021-01-04 14:09:43 +0100712 a: number, #comment
713 b: bool, # comment
714 c: string) #comment
715 assert_equal(4, a)
716 assert_equal(true, b)
717 assert_equal('yes', c)
718enddef
719
720def Test_func_with_comments()
721 FuncWithComment(4, true, 'yes')
722
723 var lines =<< trim END
724 def Func(# comment
725 arg: string)
726 enddef
727 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000728 v9.CheckScriptFailure(lines, 'E125:', 1)
Bram Moolenaarcef12702021-01-04 14:09:43 +0100729
730 lines =<< trim END
731 def Func(
732 arg: string# comment
733 )
734 enddef
735 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000736 v9.CheckScriptFailure(lines, 'E475:', 2)
Bram Moolenaarcef12702021-01-04 14:09:43 +0100737
738 lines =<< trim END
739 def Func(
740 arg: string
741 )# comment
742 enddef
743 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000744 v9.CheckScriptFailure(lines, 'E488:', 3)
Bram Moolenaarcef12702021-01-04 14:09:43 +0100745enddef
746
Bram Moolenaar04b12692020-05-04 23:24:44 +0200747def Test_nested_function()
Bram Moolenaar38453522021-11-28 22:00:12 +0000748 def NestedDef(arg: string): string
Bram Moolenaar04b12692020-05-04 23:24:44 +0200749 return 'nested ' .. arg
750 enddef
Bram Moolenaar38453522021-11-28 22:00:12 +0000751 NestedDef(':def')->assert_equal('nested :def')
752
753 func NestedFunc(arg)
754 return 'nested ' .. a:arg
755 endfunc
756 NestedFunc(':func')->assert_equal('nested :func')
Bram Moolenaar04b12692020-05-04 23:24:44 +0200757
Bram Moolenaar62aec932022-01-29 21:45:34 +0000758 v9.CheckDefFailure(['def Nested()', 'enddef', 'Nested(66)'], 'E118:')
759 v9.CheckDefFailure(['def Nested(arg: string)', 'enddef', 'Nested()'], 'E119:')
Bram Moolenaar0e65d3d2020-05-05 17:53:16 +0200760
Bram Moolenaar62aec932022-01-29 21:45:34 +0000761 v9.CheckDefFailure(['def s:Nested()', 'enddef'], 'E1075:')
762 v9.CheckDefFailure(['def b:Nested()', 'enddef'], 'E1075:')
Bram Moolenaar8b848ca2020-09-10 22:28:01 +0200763
Bram Moolenaar54021752020-12-06 18:50:36 +0100764 var lines =<< trim END
765 def Outer()
766 def Inner()
767 # comment
768 enddef
769 def Inner()
770 enddef
771 enddef
772 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000773 v9.CheckDefFailure(lines, 'E1073:')
Bram Moolenaar54021752020-12-06 18:50:36 +0100774
775 lines =<< trim END
776 def Outer()
777 def Inner()
778 # comment
779 enddef
780 def! Inner()
781 enddef
782 enddef
783 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000784 v9.CheckDefFailure(lines, 'E1117:')
Bram Moolenaar54021752020-12-06 18:50:36 +0100785
Bram Moolenaardb8e5c22021-12-25 19:58:22 +0000786 lines =<< trim END
787 vim9script
788 def Outer()
789 def Inner()
790 g:result = 'ok'
791 enddef
792 Inner()
793 enddef
794 Outer()
795 Inner()
796 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000797 v9.CheckScriptFailure(lines, 'E117: Unknown function: Inner')
Bram Moolenaardb8e5c22021-12-25 19:58:22 +0000798 assert_equal('ok', g:result)
799 unlet g:result
800
Bram Moolenaarf681cfb2022-02-07 20:30:57 +0000801 lines =<< trim END
802 vim9script
803 def Outer()
804 def _Inner()
805 echo 'bad'
806 enddef
Bram Moolenaar3787f262022-02-07 21:54:01 +0000807 _Inner()
Bram Moolenaarf681cfb2022-02-07 20:30:57 +0000808 enddef
809 defcompile
810 END
Bram Moolenaar3787f262022-02-07 21:54:01 +0000811 v9.CheckScriptFailure(lines, 'E1267:')
Bram Moolenaarf681cfb2022-02-07 20:30:57 +0000812
813 lines =<< trim END
814 vim9script
815 def Outer()
816 def g:inner()
817 echo 'bad'
818 enddef
Bram Moolenaar3787f262022-02-07 21:54:01 +0000819 g:inner()
Bram Moolenaarf681cfb2022-02-07 20:30:57 +0000820 enddef
821 defcompile
822 END
Bram Moolenaar3787f262022-02-07 21:54:01 +0000823 v9.CheckScriptFailure(lines, 'E1267:')
824
825 lines =<< trim END
826 vim9script
827 def g:_Func()
828 echo 'bad'
829 enddef
830 END
831 v9.CheckScriptFailure(lines, 'E1267:')
832
833 lines =<< trim END
834 vim9script
Bram Moolenaara749a422022-02-12 19:52:25 +0000835 def _Func()
Bram Moolenaar3787f262022-02-07 21:54:01 +0000836 echo 'bad'
837 enddef
838 END
839 v9.CheckScriptFailure(lines, 'E1267:')
Bram Moolenaarf681cfb2022-02-07 20:30:57 +0000840
Bram Moolenaar54021752020-12-06 18:50:36 +0100841 # nested function inside conditional
Bram Moolenaar54021752020-12-06 18:50:36 +0100842 lines =<< trim END
843 vim9script
844 var thecount = 0
845 if true
846 def Test(): number
847 def TheFunc(): number
848 thecount += 1
849 return thecount
850 enddef
851 return TheFunc()
852 enddef
853 endif
854 defcompile
855 assert_equal(1, Test())
856 assert_equal(2, Test())
857 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000858 v9.CheckScriptSuccess(lines)
Bram Moolenaar8863bda2021-03-17 18:42:08 +0100859
860 # also works when "thecount" is inside the "if" block
861 lines =<< trim END
862 vim9script
863 if true
864 var thecount = 0
865 def Test(): number
866 def TheFunc(): number
867 thecount += 1
868 return thecount
869 enddef
870 return TheFunc()
871 enddef
872 endif
873 defcompile
874 assert_equal(1, Test())
875 assert_equal(2, Test())
876 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000877 v9.CheckScriptSuccess(lines)
Bram Moolenaar4bba16d2021-08-15 19:28:05 +0200878
Bram Moolenaara915fa02022-03-23 11:29:15 +0000879 # nested function with recursive call
880 lines =<< trim END
881 vim9script
882
883 def MyFunc(): number
884 def Fib(n: number): number
885 if n < 2
886 return 1
887 endif
888 return Fib(n - 2) + Fib(n - 1)
889 enddef
890
891 return Fib(5)
892 enddef
893
894 assert_equal(8, MyFunc())
895 END
896 v9.CheckScriptSuccess(lines)
897
Bram Moolenaar4bba16d2021-08-15 19:28:05 +0200898 lines =<< trim END
899 vim9script
900 def Outer()
901 def Inner()
902 echo 'hello'
903 enddef burp
904 enddef
905 defcompile
906 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000907 v9.CheckScriptFailure(lines, 'E1173: Text found after enddef: burp', 3)
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200908enddef
909
Bram Moolenaaradc8e442020-12-31 18:28:18 +0100910def Test_not_nested_function()
911 echo printf('%d',
912 function('len')('xxx'))
913enddef
914
Bram Moolenaaraf8edbb2020-08-01 00:03:09 +0200915func Test_call_default_args_from_func()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +0200916 call MyDefaultArgs()->assert_equal('string')
917 call MyDefaultArgs('one')->assert_equal('one')
Bram Moolenaar9bd5d872020-09-06 21:47:48 +0200918 call assert_fails('call MyDefaultArgs("one", "two")', 'E118:', '', 3, 'Test_call_default_args_from_func')
Bram Moolenaaraf8edbb2020-08-01 00:03:09 +0200919endfunc
920
Bram Moolenaar38ddf332020-07-31 22:05:04 +0200921def Test_nested_global_function()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +0200922 var lines =<< trim END
Bram Moolenaar38ddf332020-07-31 22:05:04 +0200923 vim9script
924 def Outer()
925 def g:Inner(): string
926 return 'inner'
927 enddef
928 enddef
Bram Moolenaaraf8edbb2020-08-01 00:03:09 +0200929 defcompile
930 Outer()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +0200931 g:Inner()->assert_equal('inner')
Bram Moolenaaraf8edbb2020-08-01 00:03:09 +0200932 delfunc g:Inner
933 Outer()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +0200934 g:Inner()->assert_equal('inner')
Bram Moolenaaraf8edbb2020-08-01 00:03:09 +0200935 delfunc g:Inner
936 Outer()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +0200937 g:Inner()->assert_equal('inner')
Bram Moolenaaraf8edbb2020-08-01 00:03:09 +0200938 delfunc g:Inner
Bram Moolenaar38ddf332020-07-31 22:05:04 +0200939 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000940 v9.CheckScriptSuccess(lines)
Bram Moolenaar2c79e9d2020-08-01 18:57:52 +0200941
942 lines =<< trim END
943 vim9script
944 def Outer()
Bram Moolenaar38453522021-11-28 22:00:12 +0000945 func g:Inner()
946 return 'inner'
947 endfunc
948 enddef
949 defcompile
950 Outer()
951 g:Inner()->assert_equal('inner')
952 delfunc g:Inner
953 Outer()
954 g:Inner()->assert_equal('inner')
955 delfunc g:Inner
956 Outer()
957 g:Inner()->assert_equal('inner')
958 delfunc g:Inner
959 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000960 v9.CheckScriptSuccess(lines)
Bram Moolenaar38453522021-11-28 22:00:12 +0000961
962 lines =<< trim END
963 vim9script
964 def Outer()
Bram Moolenaar2c79e9d2020-08-01 18:57:52 +0200965 def g:Inner(): string
966 return 'inner'
967 enddef
968 enddef
969 defcompile
970 Outer()
971 Outer()
972 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000973 v9.CheckScriptFailure(lines, "E122:")
Bram Moolenaarcd45ed02020-12-22 17:35:54 +0100974 delfunc g:Inner
Bram Moolenaarad486a02020-08-01 23:22:18 +0200975
976 lines =<< trim END
977 vim9script
Bram Moolenaar58a52f22020-12-22 18:56:55 +0100978 def Outer()
979 def g:Inner()
Bram Moolenaar2949cfd2020-12-31 21:28:47 +0100980 echo map([1, 2, 3], (_, v) => v + 1)
Bram Moolenaar58a52f22020-12-22 18:56:55 +0100981 enddef
982 g:Inner()
983 enddef
984 Outer()
985 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000986 v9.CheckScriptSuccess(lines)
Bram Moolenaar58a52f22020-12-22 18:56:55 +0100987 delfunc g:Inner
988
989 lines =<< trim END
990 vim9script
Bram Moolenaarad486a02020-08-01 23:22:18 +0200991 def Func()
992 echo 'script'
993 enddef
994 def Outer()
995 def Func()
996 echo 'inner'
997 enddef
998 enddef
999 defcompile
1000 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001001 v9.CheckScriptFailure(lines, "E1073:", 1)
Bram Moolenaard604d782021-11-20 21:46:20 +00001002
1003 lines =<< trim END
1004 vim9script
1005 def Func()
1006 echo 'script'
1007 enddef
1008 def Func()
1009 echo 'script'
1010 enddef
1011 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001012 v9.CheckScriptFailure(lines, "E1073:", 5)
Bram Moolenaar38ddf332020-07-31 22:05:04 +02001013enddef
1014
Bram Moolenaar6abdcf82020-11-22 18:15:44 +01001015def DefListAll()
1016 def
1017enddef
1018
1019def DefListOne()
1020 def DefListOne
1021enddef
1022
1023def DefListMatches()
1024 def /DefList
1025enddef
1026
1027def Test_nested_def_list()
1028 var funcs = split(execute('call DefListAll()'), "\n")
1029 assert_true(len(funcs) > 10)
1030 assert_true(funcs->index('def DefListAll()') >= 0)
1031
1032 funcs = split(execute('call DefListOne()'), "\n")
1033 assert_equal([' def DefListOne()', '1 def DefListOne', ' enddef'], funcs)
1034
1035 funcs = split(execute('call DefListMatches()'), "\n")
1036 assert_true(len(funcs) >= 3)
1037 assert_true(funcs->index('def DefListAll()') >= 0)
1038 assert_true(funcs->index('def DefListOne()') >= 0)
1039 assert_true(funcs->index('def DefListMatches()') >= 0)
Bram Moolenaar54021752020-12-06 18:50:36 +01001040
1041 var lines =<< trim END
1042 vim9script
1043 def Func()
1044 def +Func+
1045 enddef
1046 defcompile
1047 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001048 v9.CheckScriptFailure(lines, 'E476:', 1)
Bram Moolenaar6abdcf82020-11-22 18:15:44 +01001049enddef
1050
Bram Moolenaare08be092022-02-17 13:08:26 +00001051def Test_global_function_not_found()
1052 var lines =<< trim END
1053 g:Ref = 123
1054 call g:Ref()
1055 END
1056 v9.CheckDefExecAndScriptFailure(lines, ['E117:', 'E1085:'], 2)
1057enddef
1058
Bram Moolenaar333894b2020-08-01 18:53:07 +02001059def Test_global_local_function()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001060 var lines =<< trim END
Bram Moolenaar333894b2020-08-01 18:53:07 +02001061 vim9script
1062 def g:Func(): string
1063 return 'global'
1064 enddef
1065 def Func(): string
1066 return 'local'
1067 enddef
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001068 g:Func()->assert_equal('global')
1069 Func()->assert_equal('local')
Bram Moolenaar2d870f82020-12-05 13:41:01 +01001070 delfunc g:Func
Bram Moolenaar333894b2020-08-01 18:53:07 +02001071 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001072 v9.CheckScriptSuccess(lines)
Bram Moolenaar035d6e92020-08-11 22:30:42 +02001073
1074 lines =<< trim END
1075 vim9script
1076 def g:Funcy()
1077 echo 'funcy'
1078 enddef
Bram Moolenaara749a422022-02-12 19:52:25 +00001079 Funcy()
Bram Moolenaar035d6e92020-08-11 22:30:42 +02001080 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001081 v9.CheckScriptFailure(lines, 'E117:')
Bram Moolenaar333894b2020-08-01 18:53:07 +02001082enddef
1083
Bram Moolenaar0f769812020-09-12 18:32:34 +02001084def Test_local_function_shadows_global()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001085 var lines =<< trim END
Bram Moolenaar0f769812020-09-12 18:32:34 +02001086 vim9script
1087 def g:Gfunc(): string
1088 return 'global'
1089 enddef
1090 def AnotherFunc(): number
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001091 var Gfunc = function('len')
Bram Moolenaar0f769812020-09-12 18:32:34 +02001092 return Gfunc('testing')
1093 enddef
1094 g:Gfunc()->assert_equal('global')
1095 AnotherFunc()->assert_equal(7)
1096 delfunc g:Gfunc
1097 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001098 v9.CheckScriptSuccess(lines)
Bram Moolenaar0f769812020-09-12 18:32:34 +02001099
1100 lines =<< trim END
1101 vim9script
1102 def g:Func(): string
1103 return 'global'
1104 enddef
1105 def AnotherFunc()
1106 g:Func = function('len')
1107 enddef
1108 AnotherFunc()
1109 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001110 v9.CheckScriptFailure(lines, 'E705:')
Bram Moolenaar0f769812020-09-12 18:32:34 +02001111 delfunc g:Func
Bram Moolenaar0865b152021-04-05 15:38:51 +02001112
Bram Moolenaar62aec932022-01-29 21:45:34 +00001113 # global function is not found with g: prefix
Bram Moolenaar0865b152021-04-05 15:38:51 +02001114 lines =<< trim END
1115 vim9script
1116 def g:Func(): string
1117 return 'global'
1118 enddef
1119 def AnotherFunc(): string
1120 return Func()
1121 enddef
1122 assert_equal('global', AnotherFunc())
Bram Moolenaar0865b152021-04-05 15:38:51 +02001123 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001124 v9.CheckScriptFailure(lines, 'E117:')
1125 delfunc g:Func
Bram Moolenaar0865b152021-04-05 15:38:51 +02001126
1127 lines =<< trim END
1128 vim9script
1129 def g:Func(): string
1130 return 'global'
1131 enddef
Bram Moolenaar848fadd2022-01-30 15:28:30 +00001132 assert_equal('global', g:Func())
Bram Moolenaar0865b152021-04-05 15:38:51 +02001133 delfunc g:Func
1134 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001135 v9.CheckScriptSuccess(lines)
Bram Moolenaar58493cf2022-01-06 12:23:30 +00001136
1137 # This does not shadow "i" which is visible only inside the for loop
1138 lines =<< trim END
1139 vim9script
1140
1141 def Foo(i: number)
1142 echo i
1143 enddef
1144
1145 for i in range(3)
1146 # Foo() is compiled here
1147 Foo(i)
1148 endfor
1149 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001150 v9.CheckScriptSuccess(lines)
Bram Moolenaar0f769812020-09-12 18:32:34 +02001151enddef
1152
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001153func TakesOneArg(arg)
1154 echo a:arg
1155endfunc
1156
1157def Test_call_wrong_args()
Bram Moolenaar62aec932022-01-29 21:45:34 +00001158 v9.CheckDefFailure(['g:TakesOneArg()'], 'E119:')
1159 v9.CheckDefFailure(['g:TakesOneArg(11, 22)'], 'E118:')
1160 v9.CheckDefFailure(['bufnr(xxx)'], 'E1001:')
1161 v9.CheckScriptFailure(['def Func(Ref: func(s: string))'], 'E475:')
Bram Moolenaaree8580e2020-08-28 17:19:07 +02001162
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001163 var lines =<< trim END
Bram Moolenaaree8580e2020-08-28 17:19:07 +02001164 vim9script
1165 def Func(s: string)
1166 echo s
1167 enddef
1168 Func([])
1169 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001170 v9.CheckScriptFailure(lines, 'E1013: Argument 1: type mismatch, expected string but got list<unknown>', 5)
Bram Moolenaarb185a402020-09-18 22:42:00 +02001171
Bram Moolenaar9a015112021-12-31 14:06:45 +00001172 # argument name declared earlier is found when declaring a function
Bram Moolenaarb185a402020-09-18 22:42:00 +02001173 lines =<< trim END
1174 vim9script
Bram Moolenaarb4893b82021-02-21 22:20:24 +01001175 var name = 'piet'
1176 def FuncOne(name: string)
Bram Moolenaar3a5988c2022-02-08 19:23:35 +00001177 echo name
Bram Moolenaarb4893b82021-02-21 22:20:24 +01001178 enddef
1179 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001180 v9.CheckScriptFailure(lines, 'E1168:')
Bram Moolenaarb4893b82021-02-21 22:20:24 +01001181
Bram Moolenaar3a5988c2022-02-08 19:23:35 +00001182 # same, inside the same block
1183 lines =<< trim END
1184 vim9script
1185 if true
1186 var name = 'piet'
1187 def FuncOne(name: string)
1188 echo name
1189 enddef
1190 endif
1191 END
1192 v9.CheckScriptFailure(lines, 'E1168:')
1193
1194 # variable in other block is OK
1195 lines =<< trim END
1196 vim9script
1197 if true
1198 var name = 'piet'
1199 endif
1200 def FuncOne(name: string)
1201 echo name
1202 enddef
1203 END
1204 v9.CheckScriptSuccess(lines)
1205
Bram Moolenaardce24412022-02-08 20:35:30 +00001206 # with another variable in another block
1207 lines =<< trim END
1208 vim9script
1209 if true
1210 var name = 'piet'
1211 # define a function so that the variable isn't cleared
1212 def GetItem(): string
1213 return item
1214 enddef
1215 endif
1216 if true
1217 var name = 'peter'
1218 def FuncOne(name: string)
1219 echo name
1220 enddef
1221 endif
1222 END
1223 v9.CheckScriptFailure(lines, 'E1168:')
1224
1225 # only variable in another block is OK
1226 lines =<< trim END
1227 vim9script
1228 if true
1229 var name = 'piet'
1230 # define a function so that the variable isn't cleared
1231 def GetItem(): string
1232 return item
1233 enddef
1234 endif
1235 if true
1236 def FuncOne(name: string)
1237 echo name
1238 enddef
1239 endif
1240 END
1241 v9.CheckScriptSuccess(lines)
1242
Bram Moolenaar9a015112021-12-31 14:06:45 +00001243 # argument name declared later is only found when compiling
1244 lines =<< trim END
1245 vim9script
1246 def FuncOne(name: string)
1247 echo nr
1248 enddef
1249 var name = 'piet'
1250 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001251 v9.CheckScriptSuccess(lines)
1252 v9.CheckScriptFailure(lines + ['defcompile'], 'E1168:')
Bram Moolenaar9a015112021-12-31 14:06:45 +00001253
Bram Moolenaarb4893b82021-02-21 22:20:24 +01001254 lines =<< trim END
1255 vim9script
Bram Moolenaarb185a402020-09-18 22:42:00 +02001256 def FuncOne(nr: number)
1257 echo nr
1258 enddef
1259 def FuncTwo()
1260 FuncOne()
1261 enddef
1262 defcompile
1263 END
1264 writefile(lines, 'Xscript')
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001265 var didCatch = false
Bram Moolenaarb185a402020-09-18 22:42:00 +02001266 try
1267 source Xscript
1268 catch
1269 assert_match('E119: Not enough arguments for function: <SNR>\d\+_FuncOne', v:exception)
1270 assert_match('Xscript\[8\]..function <SNR>\d\+_FuncTwo, line 1', v:throwpoint)
1271 didCatch = true
1272 endtry
1273 assert_true(didCatch)
1274
1275 lines =<< trim END
1276 vim9script
1277 def FuncOne(nr: number)
1278 echo nr
1279 enddef
1280 def FuncTwo()
1281 FuncOne(1, 2)
1282 enddef
1283 defcompile
1284 END
1285 writefile(lines, 'Xscript')
1286 didCatch = false
1287 try
1288 source Xscript
1289 catch
1290 assert_match('E118: Too many arguments for function: <SNR>\d\+_FuncOne', v:exception)
1291 assert_match('Xscript\[8\]..function <SNR>\d\+_FuncTwo, line 1', v:throwpoint)
1292 didCatch = true
1293 endtry
1294 assert_true(didCatch)
1295
1296 delete('Xscript')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001297enddef
1298
Bram Moolenaar50824712020-12-20 21:10:17 +01001299def Test_call_funcref_wrong_args()
1300 var head =<< trim END
1301 vim9script
1302 def Func3(a1: string, a2: number, a3: list<number>)
1303 echo a1 .. a2 .. a3[0]
1304 enddef
1305 def Testme()
1306 var funcMap: dict<func> = {func: Func3}
1307 END
1308 var tail =<< trim END
1309 enddef
1310 Testme()
1311 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001312 v9.CheckScriptSuccess(head + ["funcMap['func']('str', 123, [1, 2, 3])"] + tail)
Bram Moolenaar50824712020-12-20 21:10:17 +01001313
Bram Moolenaar62aec932022-01-29 21:45:34 +00001314 v9.CheckScriptFailure(head + ["funcMap['func']('str', 123)"] + tail, 'E119:')
1315 v9.CheckScriptFailure(head + ["funcMap['func']('str', 123, [1], 4)"] + tail, 'E118:')
Bram Moolenaar32b3f822021-01-06 21:59:39 +01001316
1317 var lines =<< trim END
1318 vim9script
1319 var Ref: func(number): any
1320 Ref = (j) => !j
1321 echo Ref(false)
1322 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001323 v9.CheckScriptFailure(lines, 'E1013: Argument 1: type mismatch, expected number but got bool', 4)
Bram Moolenaar32b3f822021-01-06 21:59:39 +01001324
1325 lines =<< trim END
1326 vim9script
1327 var Ref: func(number): any
1328 Ref = (j) => !j
1329 call Ref(false)
1330 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001331 v9.CheckScriptFailure(lines, 'E1013: Argument 1: type mismatch, expected number but got bool', 4)
Bram Moolenaar50824712020-12-20 21:10:17 +01001332enddef
1333
Bram Moolenaarb4d16cb2020-11-05 18:45:46 +01001334def Test_call_lambda_args()
Bram Moolenaar2a389082021-04-09 20:24:31 +02001335 var lines =<< trim END
1336 var Callback = (..._) => 'anything'
1337 assert_equal('anything', Callback())
1338 assert_equal('anything', Callback(1))
1339 assert_equal('anything', Callback('a', 2))
Bram Moolenaar1088b692021-04-09 22:12:44 +02001340
1341 assert_equal('xyz', ((a: string): string => a)('xyz'))
Bram Moolenaar2a389082021-04-09 20:24:31 +02001342 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001343 v9.CheckDefAndScriptSuccess(lines)
Bram Moolenaar2a389082021-04-09 20:24:31 +02001344
Bram Moolenaar62aec932022-01-29 21:45:34 +00001345 v9.CheckDefFailure(['echo ((i) => 0)()'],
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01001346 'E119: Not enough arguments for function: ((i) => 0)()')
Bram Moolenaarb4d16cb2020-11-05 18:45:46 +01001347
Bram Moolenaar2a389082021-04-09 20:24:31 +02001348 lines =<< trim END
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01001349 var Ref = (x: number, y: number) => x + y
Bram Moolenaarb4d16cb2020-11-05 18:45:46 +01001350 echo Ref(1, 'x')
1351 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001352 v9.CheckDefFailure(lines, 'E1013: Argument 2: type mismatch, expected number but got string')
Bram Moolenaare68b02a2021-01-03 13:09:51 +01001353
1354 lines =<< trim END
1355 var Ref: func(job, string, number)
1356 Ref = (x, y) => 0
1357 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001358 v9.CheckDefAndScriptFailure(lines, 'E1012:')
Bram Moolenaare68b02a2021-01-03 13:09:51 +01001359
1360 lines =<< trim END
1361 var Ref: func(job, string)
1362 Ref = (x, y, z) => 0
1363 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001364 v9.CheckDefAndScriptFailure(lines, 'E1012:')
Bram Moolenaar057e84a2021-02-28 16:55:11 +01001365
1366 lines =<< trim END
1367 var one = 1
1368 var l = [1, 2, 3]
1369 echo map(l, (one) => one)
1370 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001371 v9.CheckDefFailure(lines, 'E1167:')
1372 v9.CheckScriptFailure(['vim9script'] + lines, 'E1168:')
Bram Moolenaar057e84a2021-02-28 16:55:11 +01001373
1374 lines =<< trim END
Bram Moolenaar14ded112021-06-26 19:25:49 +02001375 var Ref: func(any, ?any): bool
1376 Ref = (_, y = 1) => false
1377 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001378 v9.CheckDefAndScriptFailure(lines, 'E1172:')
Bram Moolenaar14ded112021-06-26 19:25:49 +02001379
1380 lines =<< trim END
Bram Moolenaar015cf102021-06-26 21:52:02 +02001381 var a = 0
1382 var b = (a == 0 ? 1 : 2)
1383 assert_equal(1, b)
Bram Moolenaar98f9a5f2021-06-26 22:22:38 +02001384 var txt = 'a'
1385 b = (txt =~ 'x' ? 1 : 2)
1386 assert_equal(2, b)
Bram Moolenaar015cf102021-06-26 21:52:02 +02001387 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001388 v9.CheckDefAndScriptSuccess(lines)
Bram Moolenaar015cf102021-06-26 21:52:02 +02001389
1390 lines =<< trim END
Bram Moolenaar057e84a2021-02-28 16:55:11 +01001391 def ShadowLocal()
1392 var one = 1
1393 var l = [1, 2, 3]
1394 echo map(l, (one) => one)
1395 enddef
1396 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001397 v9.CheckDefFailure(lines, 'E1167:')
Bram Moolenaar057e84a2021-02-28 16:55:11 +01001398
1399 lines =<< trim END
1400 def Shadowarg(one: number)
1401 var l = [1, 2, 3]
1402 echo map(l, (one) => one)
1403 enddef
1404 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001405 v9.CheckDefFailure(lines, 'E1167:')
Bram Moolenaar767034c2021-04-09 17:24:52 +02001406
1407 lines =<< trim END
1408 echo ((a) => a)('aa', 'bb')
1409 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001410 v9.CheckDefAndScriptFailure(lines, 'E118:', 1)
Bram Moolenaarc4c56422021-07-21 20:38:46 +02001411
1412 lines =<< trim END
1413 echo 'aa'->((a) => a)('bb')
1414 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001415 v9.CheckDefFailure(lines, 'E118: Too many arguments for function: ->((a) => a)(''bb'')', 1)
1416 v9.CheckScriptFailure(['vim9script'] + lines, 'E118: Too many arguments for function: <lambda>', 2)
Bram Moolenaarb4d16cb2020-11-05 18:45:46 +01001417enddef
1418
Bram Moolenaara755fdb2021-11-20 21:35:41 +00001419def Test_lambda_line_nr()
1420 var lines =<< trim END
1421 vim9script
1422 # comment
1423 # comment
1424 var id = timer_start(1'000, (_) => 0)
1425 var out = execute('verbose ' .. timer_info(id)[0].callback
1426 ->string()
1427 ->substitute("('\\|')", ' ', 'g'))
1428 assert_match('Last set from .* line 4', out)
1429 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001430 v9.CheckScriptSuccess(lines)
Bram Moolenaara755fdb2021-11-20 21:35:41 +00001431enddef
1432
Bram Moolenaar5f91e742021-03-17 21:29:29 +01001433def FilterWithCond(x: string, Cond: func(string): bool): bool
1434 return Cond(x)
1435enddef
1436
Bram Moolenaar0346b792021-01-31 22:18:29 +01001437def Test_lambda_return_type()
1438 var lines =<< trim END
1439 var Ref = (): => 123
1440 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001441 v9.CheckDefAndScriptFailure(lines, 'E1157:', 1)
Bram Moolenaar5f91e742021-03-17 21:29:29 +01001442
Yegappan Lakshmanan611728f2021-05-24 15:15:47 +02001443 # no space before the return type
1444 lines =<< trim END
1445 var Ref = (x):number => x + 1
1446 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001447 v9.CheckDefAndScriptFailure(lines, 'E1069:', 1)
Yegappan Lakshmanan611728f2021-05-24 15:15:47 +02001448
Bram Moolenaar5f91e742021-03-17 21:29:29 +01001449 # this works
1450 for x in ['foo', 'boo']
Bram Moolenaar62aec932022-01-29 21:45:34 +00001451 echo g:FilterWithCond(x, (v) => v =~ '^b')
Bram Moolenaar5f91e742021-03-17 21:29:29 +01001452 endfor
1453
1454 # this fails
1455 lines =<< trim END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001456 echo g:FilterWithCond('foo', (v) => v .. '^b')
Bram Moolenaar5f91e742021-03-17 21:29:29 +01001457 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001458 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 +02001459
1460 lines =<< trim END
1461 var Lambda1 = (x) => {
1462 return x
1463 }
1464 assert_equal('asdf', Lambda1('asdf'))
1465 var Lambda2 = (x): string => {
1466 return x
1467 }
1468 assert_equal('foo', Lambda2('foo'))
1469 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001470 v9.CheckDefAndScriptSuccess(lines)
Bram Moolenaara9931532021-06-12 15:58:16 +02001471
1472 lines =<< trim END
1473 var Lambda = (x): string => {
1474 return x
1475 }
1476 echo Lambda(['foo'])
1477 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001478 v9.CheckDefExecAndScriptFailure(lines, 'E1012:')
Bram Moolenaar0346b792021-01-31 22:18:29 +01001479enddef
1480
Bram Moolenaar709664c2020-12-12 14:33:41 +01001481def Test_lambda_uses_assigned_var()
Bram Moolenaar62aec932022-01-29 21:45:34 +00001482 v9.CheckDefSuccess([
Bram Moolenaar709664c2020-12-12 14:33:41 +01001483 'var x: any = "aaa"'
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01001484 'x = filter(["bbb"], (_, v) => v =~ x)'])
Bram Moolenaar709664c2020-12-12 14:33:41 +01001485enddef
1486
Bram Moolenaar18062fc2021-03-05 21:35:47 +01001487def Test_pass_legacy_lambda_to_def_func()
1488 var lines =<< trim END
1489 vim9script
1490 func Foo()
1491 eval s:Bar({x -> 0})
1492 endfunc
1493 def Bar(y: any)
1494 enddef
1495 Foo()
1496 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001497 v9.CheckScriptSuccess(lines)
Bram Moolenaar831bdf82021-06-22 19:32:17 +02001498
1499 lines =<< trim END
1500 vim9script
Bram Moolenaar1d9cef72022-03-17 16:30:03 +00001501 def g:TestFunc(F: func)
Bram Moolenaar831bdf82021-06-22 19:32:17 +02001502 enddef
1503 legacy call g:TestFunc({-> 0})
1504 delfunc g:TestFunc
1505
Bram Moolenaar1d9cef72022-03-17 16:30:03 +00001506 def g:TestFunc(F: func(number))
Bram Moolenaar831bdf82021-06-22 19:32:17 +02001507 enddef
1508 legacy call g:TestFunc({nr -> 0})
1509 delfunc g:TestFunc
1510 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001511 v9.CheckScriptSuccess(lines)
Bram Moolenaar18062fc2021-03-05 21:35:47 +01001512enddef
1513
Bram Moolenaar844fb642021-10-23 13:32:30 +01001514def Test_lambda_in_reduce_line_break()
1515 # this was using freed memory
1516 var lines =<< trim END
1517 vim9script
1518 const result: dict<number> =
1519 ['Bob', 'Sam', 'Cat', 'Bob', 'Cat', 'Cat']
1520 ->reduce((acc, val) => {
1521 if has_key(acc, val)
1522 acc[val] += 1
1523 return acc
1524 else
1525 acc[val] = 1
1526 return acc
1527 endif
1528 }, {})
1529 assert_equal({Bob: 2, Sam: 1, Cat: 3}, result)
1530 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001531 v9.CheckScriptSuccess(lines)
Bram Moolenaar844fb642021-10-23 13:32:30 +01001532enddef
1533
Bram Moolenaardcb53be2021-12-09 14:23:43 +00001534def Test_set_opfunc_to_lambda()
1535 var lines =<< trim END
1536 vim9script
1537 nnoremap <expr> <F4> <SID>CountSpaces() .. '_'
1538 def CountSpaces(type = ''): string
1539 if type == ''
1540 &operatorfunc = (t) => CountSpaces(t)
1541 return 'g@'
1542 endif
1543 normal! '[V']y
1544 g:result = getreg('"')->count(' ')
1545 return ''
1546 enddef
1547 new
1548 'a b c d e'->setline(1)
1549 feedkeys("\<F4>", 'x')
1550 assert_equal(4, g:result)
1551 bwipe!
1552 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001553 v9.CheckScriptSuccess(lines)
Bram Moolenaardcb53be2021-12-09 14:23:43 +00001554enddef
1555
Bram Moolenaaref082e12021-12-12 21:02:03 +00001556def Test_set_opfunc_to_global_function()
1557 var lines =<< trim END
1558 vim9script
1559 def g:CountSpaces(type = ''): string
1560 normal! '[V']y
1561 g:result = getreg('"')->count(' ')
1562 return ''
1563 enddef
Bram Moolenaarb15cf442021-12-16 15:49:43 +00001564 # global function works at script level
Bram Moolenaaref082e12021-12-12 21:02:03 +00001565 &operatorfunc = g:CountSpaces
1566 new
1567 'a b c d e'->setline(1)
1568 feedkeys("g@_", 'x')
1569 assert_equal(4, g:result)
Bram Moolenaarb15cf442021-12-16 15:49:43 +00001570
1571 &operatorfunc = ''
1572 g:result = 0
1573 # global function works in :def function
1574 def Func()
1575 &operatorfunc = g:CountSpaces
1576 enddef
1577 Func()
1578 feedkeys("g@_", 'x')
1579 assert_equal(4, g:result)
1580
Bram Moolenaaref082e12021-12-12 21:02:03 +00001581 bwipe!
1582 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001583 v9.CheckScriptSuccess(lines)
Bram Moolenaaref082e12021-12-12 21:02:03 +00001584 &operatorfunc = ''
1585enddef
1586
Bram Moolenaar33b968d2021-12-13 11:31:04 +00001587def Test_use_script_func_name_with_prefix()
1588 var lines =<< trim END
1589 vim9script
Bram Moolenaara749a422022-02-12 19:52:25 +00001590 func g:Getit()
Bram Moolenaar33b968d2021-12-13 11:31:04 +00001591 return 'it'
1592 endfunc
Bram Moolenaara749a422022-02-12 19:52:25 +00001593 var Fn = g:Getit
Bram Moolenaar33b968d2021-12-13 11:31:04 +00001594 assert_equal('it', Fn())
1595 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001596 v9.CheckScriptSuccess(lines)
Bram Moolenaar33b968d2021-12-13 11:31:04 +00001597enddef
1598
Bram Moolenaardd297bc2021-12-10 10:37:38 +00001599def Test_lambda_type_allocated()
1600 # Check that unreferencing a partial using a lambda can use the variable type
1601 # after the lambda has been freed and does not leak memory.
1602 var lines =<< trim END
1603 vim9script
1604
1605 func MyomniFunc1(val, findstart, base)
1606 return a:findstart ? 0 : []
1607 endfunc
1608
1609 var Lambda = (a, b) => MyomniFunc1(19, a, b)
1610 &omnifunc = Lambda
1611 Lambda = (a, b) => MyomniFunc1(20, a, b)
1612 &omnifunc = string(Lambda)
1613 Lambda = (a, b) => strlen(a)
1614 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001615 v9.CheckScriptSuccess(lines)
Bram Moolenaardd297bc2021-12-10 10:37:38 +00001616enddef
1617
Bram Moolenaara7583c42022-05-07 21:14:05 +01001618def Test_define_lambda_in_execute()
1619 var lines =<< trim [CODE]
1620 vim9script
1621
1622 def BuildFuncMultiLine(): func
1623 var x =<< trim END
1624 g:SomeRandomFunc = (d: dict<any>) => {
1625 return d.k1 + d.k2
1626 }
1627 END
1628 execute(x)
1629 return g:SomeRandomFunc
1630 enddef
1631 var ResultPlus = BuildFuncMultiLine()
1632 assert_equal(7, ResultPlus({k1: 3, k2: 4}))
1633 [CODE]
1634 v9.CheckScriptSuccess(lines)
1635 unlet g:SomeRandomFunc
1636enddef
1637
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001638" Default arg and varargs
1639def MyDefVarargs(one: string, two = 'foo', ...rest: list<string>): string
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001640 var res = one .. ',' .. two
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001641 for s in rest
1642 res ..= ',' .. s
1643 endfor
1644 return res
1645enddef
1646
1647def Test_call_def_varargs()
Bram Moolenaar62aec932022-01-29 21:45:34 +00001648 assert_fails('g:MyDefVarargs()', 'E119:', '', 1, 'Test_call_def_varargs')
1649 g:MyDefVarargs('one')->assert_equal('one,foo')
1650 g:MyDefVarargs('one', 'two')->assert_equal('one,two')
1651 g:MyDefVarargs('one', 'two', 'three')->assert_equal('one,two,three')
1652 v9.CheckDefFailure(['g:MyDefVarargs("one", 22)'],
Bram Moolenaar77072282020-09-16 17:55:40 +02001653 'E1013: Argument 2: type mismatch, expected string but got number')
Bram Moolenaar62aec932022-01-29 21:45:34 +00001654 v9.CheckDefFailure(['g:MyDefVarargs("one", "two", 123)'],
Bram Moolenaar77072282020-09-16 17:55:40 +02001655 'E1013: Argument 3: type mismatch, expected string but got number')
Bram Moolenaar24aa48b2020-07-25 16:33:02 +02001656
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001657 var lines =<< trim END
Bram Moolenaar24aa48b2020-07-25 16:33:02 +02001658 vim9script
1659 def Func(...l: list<string>)
1660 echo l
1661 enddef
1662 Func('a', 'b', 'c')
1663 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001664 v9.CheckScriptSuccess(lines)
Bram Moolenaar24aa48b2020-07-25 16:33:02 +02001665
1666 lines =<< trim END
1667 vim9script
1668 def Func(...l: list<string>)
1669 echo l
1670 enddef
1671 Func()
1672 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001673 v9.CheckScriptSuccess(lines)
Bram Moolenaar24aa48b2020-07-25 16:33:02 +02001674
1675 lines =<< trim END
1676 vim9script
Bram Moolenaar2a389082021-04-09 20:24:31 +02001677 def Func(...l: list<any>)
Bram Moolenaar2f8cbc42020-09-16 17:22:59 +02001678 echo l
1679 enddef
1680 Func(0)
1681 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001682 v9.CheckScriptSuccess(lines)
Bram Moolenaar2f8cbc42020-09-16 17:22:59 +02001683
1684 lines =<< trim END
1685 vim9script
Bram Moolenaar2a389082021-04-09 20:24:31 +02001686 def Func(...l: any)
1687 echo l
1688 enddef
1689 Func(0)
1690 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001691 v9.CheckScriptFailure(lines, 'E1180:', 2)
Bram Moolenaar2a389082021-04-09 20:24:31 +02001692
1693 lines =<< trim END
1694 vim9script
Bram Moolenaar28022722020-09-21 22:02:49 +02001695 def Func(..._l: list<string>)
1696 echo _l
1697 enddef
1698 Func('a', 'b', 'c')
1699 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001700 v9.CheckScriptSuccess(lines)
Bram Moolenaar28022722020-09-21 22:02:49 +02001701
1702 lines =<< trim END
1703 vim9script
Bram Moolenaar24aa48b2020-07-25 16:33:02 +02001704 def Func(...l: list<string>)
1705 echo l
1706 enddef
1707 Func(1, 2, 3)
1708 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001709 v9.CheckScriptFailure(lines, 'E1013: Argument 1: type mismatch')
Bram Moolenaar24aa48b2020-07-25 16:33:02 +02001710
1711 lines =<< trim END
1712 vim9script
1713 def Func(...l: list<string>)
1714 echo l
1715 enddef
1716 Func('a', 9)
1717 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001718 v9.CheckScriptFailure(lines, 'E1013: Argument 2: type mismatch')
Bram Moolenaar24aa48b2020-07-25 16:33:02 +02001719
1720 lines =<< trim END
1721 vim9script
1722 def Func(...l: list<string>)
1723 echo l
1724 enddef
1725 Func(1, 'a')
1726 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001727 v9.CheckScriptFailure(lines, 'E1013: Argument 1: type mismatch')
Bram Moolenaar4f53b792021-02-07 15:59:49 +01001728
1729 lines =<< trim END
1730 vim9script
1731 def Func( # some comment
1732 ...l = []
1733 )
1734 echo l
1735 enddef
1736 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001737 v9.CheckScriptFailure(lines, 'E1160:')
Bram Moolenaar6ce46b92021-08-07 15:35:36 +02001738
1739 lines =<< trim END
1740 vim9script
1741 def DoIt()
1742 g:Later('')
1743 enddef
1744 defcompile
1745 def g:Later(...l: list<number>)
1746 enddef
1747 DoIt()
1748 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001749 v9.CheckScriptFailure(lines, 'E1013: Argument 1: type mismatch, expected number but got string')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001750enddef
1751
Bram Moolenaar1378fbc2020-04-11 20:50:33 +02001752let s:value = ''
1753
1754def FuncOneDefArg(opt = 'text')
1755 s:value = opt
1756enddef
1757
1758def FuncTwoDefArg(nr = 123, opt = 'text'): string
1759 return nr .. opt
1760enddef
1761
1762def FuncVarargs(...arg: list<string>): string
1763 return join(arg, ',')
1764enddef
1765
1766def Test_func_type_varargs()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001767 var RefDefArg: func(?string)
Bram Moolenaar848fadd2022-01-30 15:28:30 +00001768 RefDefArg = g:FuncOneDefArg
Bram Moolenaar1378fbc2020-04-11 20:50:33 +02001769 RefDefArg()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001770 s:value->assert_equal('text')
Bram Moolenaar1378fbc2020-04-11 20:50:33 +02001771 RefDefArg('some')
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001772 s:value->assert_equal('some')
Bram Moolenaar1378fbc2020-04-11 20:50:33 +02001773
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001774 var RefDef2Arg: func(?number, ?string): string
Bram Moolenaar848fadd2022-01-30 15:28:30 +00001775 RefDef2Arg = g:FuncTwoDefArg
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001776 RefDef2Arg()->assert_equal('123text')
1777 RefDef2Arg(99)->assert_equal('99text')
1778 RefDef2Arg(77, 'some')->assert_equal('77some')
Bram Moolenaar1378fbc2020-04-11 20:50:33 +02001779
Bram Moolenaar62aec932022-01-29 21:45:34 +00001780 v9.CheckDefFailure(['var RefWrong: func(string?)'], 'E1010:')
1781 v9.CheckDefFailure(['var RefWrong: func(?string, string)'], 'E1007:')
Bram Moolenaar1378fbc2020-04-11 20:50:33 +02001782
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001783 var RefVarargs: func(...list<string>): string
Bram Moolenaar848fadd2022-01-30 15:28:30 +00001784 RefVarargs = g:FuncVarargs
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001785 RefVarargs()->assert_equal('')
1786 RefVarargs('one')->assert_equal('one')
1787 RefVarargs('one', 'two')->assert_equal('one,two')
Bram Moolenaar1378fbc2020-04-11 20:50:33 +02001788
Bram Moolenaar62aec932022-01-29 21:45:34 +00001789 v9.CheckDefFailure(['var RefWrong: func(...list<string>, string)'], 'E110:')
1790 v9.CheckDefFailure(['var RefWrong: func(...list<string>, ?string)'], 'E110:')
Bram Moolenaar1378fbc2020-04-11 20:50:33 +02001791enddef
1792
Bram Moolenaar0b76b422020-04-07 22:05:08 +02001793" Only varargs
1794def MyVarargsOnly(...args: list<string>): string
1795 return join(args, ',')
1796enddef
1797
1798def Test_call_varargs_only()
Bram Moolenaar62aec932022-01-29 21:45:34 +00001799 g:MyVarargsOnly()->assert_equal('')
1800 g:MyVarargsOnly('one')->assert_equal('one')
1801 g:MyVarargsOnly('one', 'two')->assert_equal('one,two')
1802 v9.CheckDefFailure(['g:MyVarargsOnly(1)'], 'E1013: Argument 1: type mismatch, expected string but got number')
1803 v9.CheckDefFailure(['g:MyVarargsOnly("one", 2)'], 'E1013: Argument 2: type mismatch, expected string but got number')
Bram Moolenaar0b76b422020-04-07 22:05:08 +02001804enddef
1805
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001806def Test_using_var_as_arg()
Bram Moolenaard2939812021-12-30 17:09:05 +00001807 var lines =<< trim END
1808 def Func(x: number)
1809 var x = 234
1810 enddef
1811 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001812 v9.CheckDefFailure(lines, 'E1006:')
Bram Moolenaard2939812021-12-30 17:09:05 +00001813
1814 lines =<< trim END
1815 def Func(Ref: number)
1816 def Ref()
1817 enddef
1818 enddef
1819 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001820 v9.CheckDefFailure(lines, 'E1073:')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001821enddef
1822
Bram Moolenaar62aec932022-01-29 21:45:34 +00001823def s:DictArg(arg: dict<string>)
Bram Moolenaarcb2bdb12020-05-10 22:53:56 +02001824 arg['key'] = 'value'
1825enddef
1826
Bram Moolenaar62aec932022-01-29 21:45:34 +00001827def s:ListArg(arg: list<string>)
Bram Moolenaarcb2bdb12020-05-10 22:53:56 +02001828 arg[0] = 'value'
1829enddef
1830
1831def Test_assign_to_argument()
Bram Moolenaarf5be8cd2020-07-17 20:36:00 +02001832 # works for dict and list
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001833 var d: dict<string> = {}
Bram Moolenaarcb2bdb12020-05-10 22:53:56 +02001834 DictArg(d)
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001835 d['key']->assert_equal('value')
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001836 var l: list<string> = []
Bram Moolenaarcb2bdb12020-05-10 22:53:56 +02001837 ListArg(l)
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001838 l[0]->assert_equal('value')
Bram Moolenaarcb2bdb12020-05-10 22:53:56 +02001839
Bram Moolenaar62aec932022-01-29 21:45:34 +00001840 v9.CheckScriptFailure(['def Func(arg: number)', 'arg = 3', 'enddef', 'defcompile'], 'E1090:')
Bram Moolenaar2d870f82020-12-05 13:41:01 +01001841 delfunc! g:Func
Bram Moolenaarcb2bdb12020-05-10 22:53:56 +02001842enddef
1843
Bram Moolenaarb816dae2020-09-20 22:04:00 +02001844" These argument names are reserved in legacy functions.
Bram Moolenaar62aec932022-01-29 21:45:34 +00001845def s:WithReservedNames(firstline: string, lastline: string): string
Bram Moolenaarb816dae2020-09-20 22:04:00 +02001846 return firstline .. lastline
1847enddef
1848
1849def Test_argument_names()
1850 assert_equal('OK', WithReservedNames('O', 'K'))
1851enddef
1852
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001853def Test_call_func_defined_later()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001854 g:DefinedLater('one')->assert_equal('one')
Bram Moolenaar9bd5d872020-09-06 21:47:48 +02001855 assert_fails('NotDefined("one")', 'E117:', '', 2, 'Test_call_func_defined_later')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001856enddef
1857
Bram Moolenaar1df8b3f2020-04-23 18:13:23 +02001858func DefinedLater(arg)
1859 return a:arg
1860endfunc
1861
1862def Test_call_funcref()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001863 g:SomeFunc('abc')->assert_equal(3)
Bram Moolenaar9bd5d872020-09-06 21:47:48 +02001864 assert_fails('NotAFunc()', 'E117:', '', 2, 'Test_call_funcref') # comment after call
Bram Moolenaar2ef91562021-12-11 16:14:07 +00001865 assert_fails('g:NotAFunc()', 'E1085:', '', 3, 'Test_call_funcref')
Bram Moolenaar2f1980f2020-07-22 19:30:06 +02001866
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001867 var lines =<< trim END
Bram Moolenaar2f1980f2020-07-22 19:30:06 +02001868 vim9script
1869 def RetNumber(): number
1870 return 123
1871 enddef
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001872 var Funcref: func: number = function('RetNumber')
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001873 Funcref()->assert_equal(123)
Bram Moolenaar2f1980f2020-07-22 19:30:06 +02001874 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001875 v9.CheckScriptSuccess(lines)
Bram Moolenaar0f60e802020-07-22 20:16:11 +02001876
1877 lines =<< trim END
1878 vim9script
1879 def RetNumber(): number
1880 return 123
1881 enddef
1882 def Bar(F: func: number): number
1883 return F()
1884 enddef
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001885 var Funcref = function('RetNumber')
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001886 Bar(Funcref)->assert_equal(123)
Bram Moolenaar0f60e802020-07-22 20:16:11 +02001887 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001888 v9.CheckScriptSuccess(lines)
Bram Moolenaarbfba8652020-07-23 20:09:10 +02001889
1890 lines =<< trim END
1891 vim9script
1892 def UseNumber(nr: number)
1893 echo nr
1894 enddef
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001895 var Funcref: func(number) = function('UseNumber')
Bram Moolenaarbfba8652020-07-23 20:09:10 +02001896 Funcref(123)
1897 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001898 v9.CheckScriptSuccess(lines)
Bram Moolenaarb8070e32020-07-23 20:56:04 +02001899
1900 lines =<< trim END
1901 vim9script
1902 def UseNumber(nr: number)
1903 echo nr
1904 enddef
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001905 var Funcref: func(string) = function('UseNumber')
Bram Moolenaarb8070e32020-07-23 20:56:04 +02001906 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001907 v9.CheckScriptFailure(lines, 'E1012: Type mismatch; expected func(string) but got func(number)')
Bram Moolenaar4fc224c2020-07-26 17:56:25 +02001908
1909 lines =<< trim END
1910 vim9script
1911 def EchoNr(nr = 34)
1912 g:echo = nr
1913 enddef
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001914 var Funcref: func(?number) = function('EchoNr')
Bram Moolenaar4fc224c2020-07-26 17:56:25 +02001915 Funcref()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001916 g:echo->assert_equal(34)
Bram Moolenaar4fc224c2020-07-26 17:56:25 +02001917 Funcref(123)
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001918 g:echo->assert_equal(123)
Bram Moolenaar4fc224c2020-07-26 17:56:25 +02001919 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001920 v9.CheckScriptSuccess(lines)
Bram Moolenaarace61322020-07-26 18:16:58 +02001921
1922 lines =<< trim END
1923 vim9script
1924 def EchoList(...l: list<number>)
1925 g:echo = l
1926 enddef
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001927 var Funcref: func(...list<number>) = function('EchoList')
Bram Moolenaarace61322020-07-26 18:16:58 +02001928 Funcref()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001929 g:echo->assert_equal([])
Bram Moolenaarace61322020-07-26 18:16:58 +02001930 Funcref(1, 2, 3)
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001931 g:echo->assert_equal([1, 2, 3])
Bram Moolenaarace61322020-07-26 18:16:58 +02001932 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001933 v9.CheckScriptSuccess(lines)
Bram Moolenaar01865ad2020-07-26 18:33:09 +02001934
1935 lines =<< trim END
1936 vim9script
1937 def OptAndVar(nr: number, opt = 12, ...l: list<number>): number
1938 g:optarg = opt
1939 g:listarg = l
1940 return nr
1941 enddef
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001942 var Funcref: func(number, ?number, ...list<number>): number = function('OptAndVar')
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001943 Funcref(10)->assert_equal(10)
1944 g:optarg->assert_equal(12)
1945 g:listarg->assert_equal([])
Bram Moolenaar01865ad2020-07-26 18:33:09 +02001946
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001947 Funcref(11, 22)->assert_equal(11)
1948 g:optarg->assert_equal(22)
1949 g:listarg->assert_equal([])
Bram Moolenaar01865ad2020-07-26 18:33:09 +02001950
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001951 Funcref(17, 18, 1, 2, 3)->assert_equal(17)
1952 g:optarg->assert_equal(18)
1953 g:listarg->assert_equal([1, 2, 3])
Bram Moolenaar01865ad2020-07-26 18:33:09 +02001954 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001955 v9.CheckScriptSuccess(lines)
Bram Moolenaar1df8b3f2020-04-23 18:13:23 +02001956enddef
1957
1958let SomeFunc = function('len')
1959let NotAFunc = 'text'
1960
Bram Moolenaar99aaf0c2020-04-12 14:39:53 +02001961def CombineFuncrefTypes()
Bram Moolenaarf5be8cd2020-07-17 20:36:00 +02001962 # same arguments, different return type
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001963 var Ref1: func(bool): string
1964 var Ref2: func(bool): number
1965 var Ref3: func(bool): any
Bram Moolenaar99aaf0c2020-04-12 14:39:53 +02001966 Ref3 = g:cond ? Ref1 : Ref2
1967
Bram Moolenaarf5be8cd2020-07-17 20:36:00 +02001968 # different number of arguments
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001969 var Refa1: func(bool): number
1970 var Refa2: func(bool, number): number
1971 var Refa3: func: number
Bram Moolenaar99aaf0c2020-04-12 14:39:53 +02001972 Refa3 = g:cond ? Refa1 : Refa2
1973
Bram Moolenaarf5be8cd2020-07-17 20:36:00 +02001974 # different argument types
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001975 var Refb1: func(bool, string): number
1976 var Refb2: func(string, number): number
1977 var Refb3: func(any, any): number
Bram Moolenaar99aaf0c2020-04-12 14:39:53 +02001978 Refb3 = g:cond ? Refb1 : Refb2
1979enddef
1980
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001981def FuncWithForwardCall()
Bram Moolenaar1df8b3f2020-04-23 18:13:23 +02001982 return g:DefinedEvenLater("yes")
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001983enddef
1984
1985def DefinedEvenLater(arg: string): string
1986 return arg
1987enddef
1988
1989def Test_error_in_nested_function()
Bram Moolenaarf5be8cd2020-07-17 20:36:00 +02001990 # Error in called function requires unwinding the call stack.
Bram Moolenaar848fadd2022-01-30 15:28:30 +00001991 assert_fails('g:FuncWithForwardCall()', 'E1096:', '', 1, 'FuncWithForwardCall')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001992enddef
1993
Bram Moolenaar4bf10062021-12-28 17:23:12 +00001994def Test_nested_function_with_nextcmd()
Bram Moolenaar9c23f9b2021-12-26 14:23:22 +00001995 var lines =<< trim END
1996 vim9script
1997 # Define an outer function
1998 def FirstFunction()
1999 # Define an inner function
2000 def SecondFunction()
2001 # the function has a body, a double free is detected.
2002 AAAAA
2003
2004 # enddef followed by | or } followed by # one or more characters
2005 enddef|BBBB
2006 enddef
2007
2008 # Compile all functions
2009 defcompile
2010 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002011 v9.CheckScriptFailure(lines, 'E1173: Text found after enddef: BBBB')
Bram Moolenaar9c23f9b2021-12-26 14:23:22 +00002012enddef
2013
Bram Moolenaar4bf10062021-12-28 17:23:12 +00002014def Test_nested_function_with_args_split()
2015 var lines =<< trim END
2016 vim9script
2017 def FirstFunction()
2018 def SecondFunction(
2019 )
2020 # had a double free if the right parenthesis of the nested function is
2021 # on the next line
2022
2023 enddef|BBBB
2024 enddef
2025 # Compile all functions
2026 defcompile
2027 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002028 v9.CheckScriptFailure(lines, 'E1173: Text found after enddef: BBBB')
Bram Moolenaar7473a842021-12-28 17:55:26 +00002029
2030 lines =<< trim END
2031 vim9script
2032 def FirstFunction()
2033 func SecondFunction()
2034 endfunc|BBBB
2035 enddef
2036 defcompile
2037 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002038 v9.CheckScriptFailure(lines, 'E1173: Text found after endfunction: BBBB')
Bram Moolenaar4bf10062021-12-28 17:23:12 +00002039enddef
2040
Bram Moolenaar9f1a39a2022-01-08 15:39:39 +00002041def Test_error_in_function_args()
2042 var lines =<< trim END
2043 def FirstFunction()
2044 def SecondFunction(J =
2045 # Nois
2046 # one
2047
2048 enddef|BBBB
2049 enddef
2050 # Compile all functions
2051 defcompile
2052 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002053 v9.CheckScriptFailure(lines, 'E488:')
Bram Moolenaar9f1a39a2022-01-08 15:39:39 +00002054enddef
2055
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002056def Test_return_type_wrong()
Bram Moolenaar62aec932022-01-29 21:45:34 +00002057 v9.CheckScriptFailure([
Bram Moolenaar5a849da2020-08-08 16:47:30 +02002058 'def Func(): number',
2059 'return "a"',
2060 'enddef',
2061 'defcompile'], 'expected number but got string')
Bram Moolenaar2d870f82020-12-05 13:41:01 +01002062 delfunc! g:Func
Bram Moolenaar62aec932022-01-29 21:45:34 +00002063 v9.CheckScriptFailure([
Bram Moolenaar5a849da2020-08-08 16:47:30 +02002064 'def Func(): string',
2065 'return 1',
2066 'enddef',
2067 'defcompile'], 'expected string but got number')
Bram Moolenaar2d870f82020-12-05 13:41:01 +01002068 delfunc! g:Func
Bram Moolenaar62aec932022-01-29 21:45:34 +00002069 v9.CheckScriptFailure([
Bram Moolenaar5a849da2020-08-08 16:47:30 +02002070 'def Func(): void',
2071 'return "a"',
2072 'enddef',
2073 'defcompile'],
2074 'E1096: Returning a value in a function without a return type')
Bram Moolenaar2d870f82020-12-05 13:41:01 +01002075 delfunc! g:Func
Bram Moolenaar62aec932022-01-29 21:45:34 +00002076 v9.CheckScriptFailure([
Bram Moolenaar5a849da2020-08-08 16:47:30 +02002077 'def Func()',
2078 'return "a"',
2079 'enddef',
2080 'defcompile'],
2081 'E1096: Returning a value in a function without a return type')
Bram Moolenaar2d870f82020-12-05 13:41:01 +01002082 delfunc! g:Func
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002083
Bram Moolenaar62aec932022-01-29 21:45:34 +00002084 v9.CheckScriptFailure([
Bram Moolenaar5a849da2020-08-08 16:47:30 +02002085 'def Func(): number',
2086 'return',
2087 'enddef',
2088 'defcompile'], 'E1003:')
Bram Moolenaar2d870f82020-12-05 13:41:01 +01002089 delfunc! g:Func
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002090
Bram Moolenaar62aec932022-01-29 21:45:34 +00002091 v9.CheckScriptFailure([
Bram Moolenaar33ea9fd2021-08-08 19:07:37 +02002092 'def Func():number',
2093 'return 123',
2094 'enddef',
2095 'defcompile'], 'E1069:')
2096 delfunc! g:Func
2097
Bram Moolenaar62aec932022-01-29 21:45:34 +00002098 v9.CheckScriptFailure([
Bram Moolenaar33ea9fd2021-08-08 19:07:37 +02002099 'def Func() :number',
2100 'return 123',
2101 'enddef',
2102 'defcompile'], 'E1059:')
2103 delfunc! g:Func
2104
Bram Moolenaar62aec932022-01-29 21:45:34 +00002105 v9.CheckScriptFailure([
Bram Moolenaar33ea9fd2021-08-08 19:07:37 +02002106 'def Func() : number',
2107 'return 123',
2108 'enddef',
2109 'defcompile'], 'E1059:')
2110 delfunc! g:Func
2111
Bram Moolenaar62aec932022-01-29 21:45:34 +00002112 v9.CheckScriptFailure(['def Func(): list', 'return []', 'enddef'], 'E1008:')
Bram Moolenaar2d870f82020-12-05 13:41:01 +01002113 delfunc! g:Func
Bram Moolenaar62aec932022-01-29 21:45:34 +00002114 v9.CheckScriptFailure(['def Func(): dict', 'return {}', 'enddef'], 'E1008:')
Bram Moolenaar2d870f82020-12-05 13:41:01 +01002115 delfunc! g:Func
Bram Moolenaar62aec932022-01-29 21:45:34 +00002116 v9.CheckScriptFailure(['def Func()', 'return 1'], 'E1057:')
Bram Moolenaar2d870f82020-12-05 13:41:01 +01002117 delfunc! g:Func
Bram Moolenaar5a849da2020-08-08 16:47:30 +02002118
Bram Moolenaar62aec932022-01-29 21:45:34 +00002119 v9.CheckScriptFailure([
Bram Moolenaar5a849da2020-08-08 16:47:30 +02002120 'vim9script',
2121 'def FuncB()',
2122 ' return 123',
2123 'enddef',
2124 'def FuncA()',
2125 ' FuncB()',
2126 'enddef',
2127 'defcompile'], 'E1096:')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002128enddef
2129
2130def Test_arg_type_wrong()
Bram Moolenaar62aec932022-01-29 21:45:34 +00002131 v9.CheckScriptFailure(['def Func3(items: list)', 'echo "a"', 'enddef'], 'E1008: Missing <type>')
2132 v9.CheckScriptFailure(['def Func4(...)', 'echo "a"', 'enddef'], 'E1055: Missing name after ...')
2133 v9.CheckScriptFailure(['def Func5(items:string)', 'echo "a"'], 'E1069:')
2134 v9.CheckScriptFailure(['def Func5(items)', 'echo "a"'], 'E1077:')
2135 v9.CheckScriptFailure(['def Func6(...x:list<number>)', 'echo "a"', 'enddef'], 'E1069:')
2136 v9.CheckScriptFailure(['def Func7(...x: int)', 'echo "a"', 'enddef'], 'E1010:')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002137enddef
2138
Bram Moolenaar86cdb8a2021-04-06 19:01:03 +02002139def Test_white_space_before_comma()
2140 var lines =<< trim END
2141 vim9script
2142 def Func(a: number , b: number)
2143 enddef
2144 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002145 v9.CheckScriptFailure(lines, 'E1068:')
Yegappan Lakshmanan611728f2021-05-24 15:15:47 +02002146 call assert_fails('vim9cmd echo stridx("a" .. "b" , "a")', 'E1068:')
Bram Moolenaar86cdb8a2021-04-06 19:01:03 +02002147enddef
2148
Bram Moolenaar608d78f2021-03-06 22:33:12 +01002149def Test_white_space_after_comma()
2150 var lines =<< trim END
2151 vim9script
2152 def Func(a: number,b: number)
2153 enddef
2154 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002155 v9.CheckScriptFailure(lines, 'E1069:')
Bram Moolenaar608d78f2021-03-06 22:33:12 +01002156
2157 # OK in legacy function
2158 lines =<< trim END
2159 vim9script
2160 func Func(a,b)
2161 endfunc
2162 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002163 v9.CheckScriptSuccess(lines)
Bram Moolenaar608d78f2021-03-06 22:33:12 +01002164enddef
2165
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002166def Test_vim9script_call()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002167 var lines =<< trim END
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002168 vim9script
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002169 var name = ''
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002170 def MyFunc(arg: string)
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002171 name = arg
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002172 enddef
2173 MyFunc('foobar')
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002174 name->assert_equal('foobar')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002175
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002176 var str = 'barfoo'
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002177 str->MyFunc()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002178 name->assert_equal('barfoo')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002179
Bram Moolenaar67979662020-06-20 22:50:47 +02002180 g:value = 'value'
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002181 g:value->MyFunc()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002182 name->assert_equal('value')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002183
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002184 var listvar = []
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002185 def ListFunc(arg: list<number>)
2186 listvar = arg
2187 enddef
2188 [1, 2, 3]->ListFunc()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002189 listvar->assert_equal([1, 2, 3])
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002190
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002191 var dictvar = {}
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002192 def DictFunc(arg: dict<number>)
2193 dictvar = arg
2194 enddef
Bram Moolenaare0de1712020-12-02 17:36:54 +01002195 {a: 1, b: 2}->DictFunc()
2196 dictvar->assert_equal({a: 1, b: 2})
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002197 def CompiledDict()
Bram Moolenaare0de1712020-12-02 17:36:54 +01002198 {a: 3, b: 4}->DictFunc()
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002199 enddef
2200 CompiledDict()
Bram Moolenaare0de1712020-12-02 17:36:54 +01002201 dictvar->assert_equal({a: 3, b: 4})
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002202
Bram Moolenaare0de1712020-12-02 17:36:54 +01002203 {a: 3, b: 4}->DictFunc()
2204 dictvar->assert_equal({a: 3, b: 4})
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002205
2206 ('text')->MyFunc()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002207 name->assert_equal('text')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002208 ("some")->MyFunc()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002209 name->assert_equal('some')
Bram Moolenaare6b53242020-07-01 17:28:33 +02002210
Bram Moolenaar13e12b82020-07-24 18:47:22 +02002211 # line starting with single quote is not a mark
Bram Moolenaar10409562020-07-29 20:00:38 +02002212 # line starting with double quote can be a method call
Bram Moolenaar3d48e252020-07-15 14:15:52 +02002213 'asdfasdf'->MyFunc()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002214 name->assert_equal('asdfasdf')
Bram Moolenaar10409562020-07-29 20:00:38 +02002215 "xyz"->MyFunc()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002216 name->assert_equal('xyz')
Bram Moolenaar3d48e252020-07-15 14:15:52 +02002217
2218 def UseString()
2219 'xyork'->MyFunc()
2220 enddef
2221 UseString()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002222 name->assert_equal('xyork')
Bram Moolenaar3d48e252020-07-15 14:15:52 +02002223
Bram Moolenaar10409562020-07-29 20:00:38 +02002224 def UseString2()
2225 "knife"->MyFunc()
2226 enddef
2227 UseString2()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002228 name->assert_equal('knife')
Bram Moolenaar10409562020-07-29 20:00:38 +02002229
Bram Moolenaar13e12b82020-07-24 18:47:22 +02002230 # prepending a colon makes it a mark
2231 new
2232 setline(1, ['aaa', 'bbb', 'ccc'])
2233 normal! 3Gmt1G
2234 :'t
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002235 getcurpos()[1]->assert_equal(3)
Bram Moolenaar13e12b82020-07-24 18:47:22 +02002236 bwipe!
2237
Bram Moolenaare6b53242020-07-01 17:28:33 +02002238 MyFunc(
2239 'continued'
2240 )
2241 assert_equal('continued',
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002242 name
Bram Moolenaare6b53242020-07-01 17:28:33 +02002243 )
2244
2245 call MyFunc(
2246 'more'
2247 ..
2248 'lines'
2249 )
2250 assert_equal(
2251 'morelines',
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002252 name)
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002253 END
2254 writefile(lines, 'Xcall.vim')
2255 source Xcall.vim
2256 delete('Xcall.vim')
2257enddef
2258
2259def Test_vim9script_call_fail_decl()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002260 var lines =<< trim END
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002261 vim9script
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002262 var name = ''
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002263 def MyFunc(arg: string)
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002264 var name = 123
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002265 enddef
Bram Moolenaar822ba242020-05-24 23:00:18 +02002266 defcompile
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002267 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002268 v9.CheckScriptFailure(lines, 'E1054:')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002269enddef
2270
Bram Moolenaar65b95452020-07-19 14:03:09 +02002271def Test_vim9script_call_fail_type()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002272 var lines =<< trim END
Bram Moolenaar65b95452020-07-19 14:03:09 +02002273 vim9script
2274 def MyFunc(arg: string)
2275 echo arg
2276 enddef
2277 MyFunc(1234)
2278 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002279 v9.CheckScriptFailure(lines, 'E1013: Argument 1: type mismatch, expected string but got number')
Bram Moolenaar65b95452020-07-19 14:03:09 +02002280enddef
2281
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002282def Test_vim9script_call_fail_const()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002283 var lines =<< trim END
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002284 vim9script
2285 const var = ''
2286 def MyFunc(arg: string)
2287 var = 'asdf'
2288 enddef
Bram Moolenaar822ba242020-05-24 23:00:18 +02002289 defcompile
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002290 END
2291 writefile(lines, 'Xcall_const.vim')
Bram Moolenaar9bd5d872020-09-06 21:47:48 +02002292 assert_fails('source Xcall_const.vim', 'E46:', '', 1, 'MyFunc')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002293 delete('Xcall_const.vim')
Bram Moolenaar3bdc90b2020-12-22 20:35:40 +01002294
2295 lines =<< trim END
2296 const g:Aconst = 77
2297 def Change()
2298 # comment
2299 g:Aconst = 99
2300 enddef
2301 call Change()
2302 unlet g:Aconst
2303 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002304 v9.CheckScriptFailure(lines, 'E741: Value is locked: Aconst', 2)
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002305enddef
2306
2307" Test that inside :function a Python function can be defined, :def is not
2308" recognized.
2309func Test_function_python()
2310 CheckFeature python3
Bram Moolenaar727345e2020-09-27 23:33:59 +02002311 let py = 'python3'
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002312 execute py "<< EOF"
2313def do_something():
2314 return 1
2315EOF
2316endfunc
2317
2318def Test_delfunc()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002319 var lines =<< trim END
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002320 vim9script
Bram Moolenaar4c17ad92020-04-27 22:47:51 +02002321 def g:GoneSoon()
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002322 echo 'hello'
2323 enddef
2324
2325 def CallGoneSoon()
Bram Moolenaar62aec932022-01-29 21:45:34 +00002326 g:GoneSoon()
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002327 enddef
Bram Moolenaar822ba242020-05-24 23:00:18 +02002328 defcompile
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002329
Bram Moolenaar4c17ad92020-04-27 22:47:51 +02002330 delfunc g:GoneSoon
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002331 CallGoneSoon()
2332 END
2333 writefile(lines, 'XToDelFunc')
Bram Moolenaar9bd5d872020-09-06 21:47:48 +02002334 assert_fails('so XToDelFunc', 'E933:', '', 1, 'CallGoneSoon')
2335 assert_fails('so XToDelFunc', 'E933:', '', 1, 'CallGoneSoon')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002336
2337 delete('XToDelFunc')
2338enddef
2339
Bram Moolenaar7509ad82021-12-14 18:14:37 +00002340func Test_free_dict_while_in_funcstack()
2341 " relies on the sleep command
2342 CheckUnix
2343 call Run_Test_free_dict_while_in_funcstack()
2344endfunc
2345
2346def Run_Test_free_dict_while_in_funcstack()
Bram Moolenaar7509ad82021-12-14 18:14:37 +00002347 # this was freeing the TermRun() default argument dictionary while it was
2348 # still referenced in a funcstack_T
2349 var lines =<< trim END
2350 vim9script
2351
2352 &updatetime = 400
2353 def TermRun(_ = {})
2354 def Post()
2355 enddef
2356 def Exec()
2357 term_start('sleep 1', {
2358 term_finish: 'close',
2359 exit_cb: (_, _) => Post(),
2360 })
2361 enddef
2362 Exec()
2363 enddef
2364 nnoremap <F4> <Cmd>call <SID>TermRun()<CR>
2365 timer_start(100, (_) => feedkeys("\<F4>"))
2366 timer_start(1000, (_) => feedkeys("\<F4>"))
2367 sleep 1500m
2368 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002369 v9.CheckScriptSuccess(lines)
Bram Moolenaar7509ad82021-12-14 18:14:37 +00002370 nunmap <F4>
2371 set updatetime&
2372enddef
2373
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002374def Test_redef_failure()
Bram Moolenaard2c61702020-09-06 15:58:36 +02002375 writefile(['def Func0(): string', 'return "Func0"', 'enddef'], 'Xdef')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002376 so Xdef
Bram Moolenaard2c61702020-09-06 15:58:36 +02002377 writefile(['def Func1(): string', 'return "Func1"', 'enddef'], 'Xdef')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002378 so Xdef
Bram Moolenaard2c61702020-09-06 15:58:36 +02002379 writefile(['def! Func0(): string', 'enddef', 'defcompile'], 'Xdef')
Bram Moolenaar9bd5d872020-09-06 21:47:48 +02002380 assert_fails('so Xdef', 'E1027:', '', 1, 'Func0')
Bram Moolenaard2c61702020-09-06 15:58:36 +02002381 writefile(['def Func2(): string', 'return "Func2"', 'enddef'], 'Xdef')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002382 so Xdef
Bram Moolenaard2c61702020-09-06 15:58:36 +02002383 delete('Xdef')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002384
Bram Moolenaar701cc6c2021-04-10 13:33:48 +02002385 assert_fails('g:Func0()', 'E1091:')
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002386 g:Func1()->assert_equal('Func1')
2387 g:Func2()->assert_equal('Func2')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002388
2389 delfunc! Func0
2390 delfunc! Func1
2391 delfunc! Func2
2392enddef
2393
Bram Moolenaarf93c7fe2020-04-23 22:16:53 +02002394def Test_vim9script_func()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002395 var lines =<< trim END
Bram Moolenaarf93c7fe2020-04-23 22:16:53 +02002396 vim9script
2397 func Func(arg)
2398 echo a:arg
2399 endfunc
2400 Func('text')
2401 END
2402 writefile(lines, 'XVim9Func')
2403 so XVim9Func
2404
2405 delete('XVim9Func')
2406enddef
2407
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002408let s:funcResult = 0
2409
2410def FuncNoArgNoRet()
Bram Moolenaar53900992020-08-22 19:02:02 +02002411 s:funcResult = 11
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002412enddef
2413
2414def FuncNoArgRetNumber(): number
Bram Moolenaar53900992020-08-22 19:02:02 +02002415 s:funcResult = 22
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002416 return 1234
2417enddef
2418
Bram Moolenaarec5929d2020-04-07 20:53:39 +02002419def FuncNoArgRetString(): string
Bram Moolenaar53900992020-08-22 19:02:02 +02002420 s:funcResult = 45
Bram Moolenaarec5929d2020-04-07 20:53:39 +02002421 return 'text'
2422enddef
2423
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002424def FuncOneArgNoRet(arg: number)
Bram Moolenaar53900992020-08-22 19:02:02 +02002425 s:funcResult = arg
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002426enddef
2427
2428def FuncOneArgRetNumber(arg: number): number
Bram Moolenaar53900992020-08-22 19:02:02 +02002429 s:funcResult = arg
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002430 return arg
2431enddef
2432
Bram Moolenaar08938ee2020-04-11 23:17:17 +02002433def FuncTwoArgNoRet(one: bool, two: number)
Bram Moolenaar53900992020-08-22 19:02:02 +02002434 s:funcResult = two
Bram Moolenaar08938ee2020-04-11 23:17:17 +02002435enddef
2436
Bram Moolenaar62aec932022-01-29 21:45:34 +00002437def s:FuncOneArgRetString(arg: string): string
Bram Moolenaarec5929d2020-04-07 20:53:39 +02002438 return arg
2439enddef
2440
Bram Moolenaar62aec932022-01-29 21:45:34 +00002441def s:FuncOneArgRetAny(arg: any): any
Bram Moolenaar89228602020-04-05 22:14:54 +02002442 return arg
2443enddef
2444
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002445def Test_func_type()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002446 var Ref1: func()
Bram Moolenaar53900992020-08-22 19:02:02 +02002447 s:funcResult = 0
Bram Moolenaar62aec932022-01-29 21:45:34 +00002448 Ref1 = g:FuncNoArgNoRet
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002449 Ref1()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002450 s:funcResult->assert_equal(11)
Bram Moolenaar4c683752020-04-05 21:38:23 +02002451
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002452 var Ref2: func
Bram Moolenaar53900992020-08-22 19:02:02 +02002453 s:funcResult = 0
Bram Moolenaar62aec932022-01-29 21:45:34 +00002454 Ref2 = g:FuncNoArgNoRet
Bram Moolenaar4c683752020-04-05 21:38:23 +02002455 Ref2()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002456 s:funcResult->assert_equal(11)
Bram Moolenaar4c683752020-04-05 21:38:23 +02002457
Bram Moolenaar53900992020-08-22 19:02:02 +02002458 s:funcResult = 0
Bram Moolenaar62aec932022-01-29 21:45:34 +00002459 Ref2 = g:FuncOneArgNoRet
Bram Moolenaar4c683752020-04-05 21:38:23 +02002460 Ref2(12)
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002461 s:funcResult->assert_equal(12)
Bram Moolenaar4c683752020-04-05 21:38:23 +02002462
Bram Moolenaar53900992020-08-22 19:02:02 +02002463 s:funcResult = 0
Bram Moolenaar62aec932022-01-29 21:45:34 +00002464 Ref2 = g:FuncNoArgRetNumber
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002465 Ref2()->assert_equal(1234)
2466 s:funcResult->assert_equal(22)
Bram Moolenaar4c683752020-04-05 21:38:23 +02002467
Bram Moolenaar53900992020-08-22 19:02:02 +02002468 s:funcResult = 0
Bram Moolenaar848fadd2022-01-30 15:28:30 +00002469 Ref2 = g:FuncOneArgRetNumber
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002470 Ref2(13)->assert_equal(13)
2471 s:funcResult->assert_equal(13)
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002472enddef
2473
Bram Moolenaar9978d472020-07-05 16:01:56 +02002474def Test_repeat_return_type()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002475 var res = 0
Bram Moolenaar9978d472020-07-05 16:01:56 +02002476 for n in repeat([1], 3)
2477 res += n
2478 endfor
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002479 res->assert_equal(3)
Bram Moolenaarfce82b32020-07-05 16:07:21 +02002480
2481 res = 0
2482 for n in add([1, 2], 3)
2483 res += n
2484 endfor
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002485 res->assert_equal(6)
Bram Moolenaar9978d472020-07-05 16:01:56 +02002486enddef
2487
Bram Moolenaar846178a2020-07-05 17:04:13 +02002488def Test_argv_return_type()
2489 next fileone filetwo
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002490 var res = ''
Bram Moolenaar846178a2020-07-05 17:04:13 +02002491 for name in argv()
2492 res ..= name
2493 endfor
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002494 res->assert_equal('fileonefiletwo')
Bram Moolenaar846178a2020-07-05 17:04:13 +02002495enddef
2496
Bram Moolenaarec5929d2020-04-07 20:53:39 +02002497def Test_func_type_part()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002498 var RefVoid: func: void
Bram Moolenaar62aec932022-01-29 21:45:34 +00002499 RefVoid = g:FuncNoArgNoRet
2500 RefVoid = g:FuncOneArgNoRet
2501 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 +00002502 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 +02002503
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002504 var RefAny: func(): any
Bram Moolenaar62aec932022-01-29 21:45:34 +00002505 RefAny = g:FuncNoArgRetNumber
Bram Moolenaar848fadd2022-01-30 15:28:30 +00002506 RefAny = g:FuncNoArgRetString
Bram Moolenaar62aec932022-01-29 21:45:34 +00002507 v9.CheckDefFailure(['var RefAny: func(): any', 'RefAny = g:FuncNoArgNoRet'], 'E1012: Type mismatch; expected func(): any but got func()')
2508 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 +02002509
Bram Moolenaar6abd3dc2020-10-04 14:17:32 +02002510 var RefAnyNoArgs: func: any = RefAny
2511
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002512 var RefNr: func: number
Bram Moolenaar62aec932022-01-29 21:45:34 +00002513 RefNr = g:FuncNoArgRetNumber
Bram Moolenaar848fadd2022-01-30 15:28:30 +00002514 RefNr = g:FuncOneArgRetNumber
Bram Moolenaar62aec932022-01-29 21:45:34 +00002515 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 +00002516 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 +02002517
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002518 var RefStr: func: string
Bram Moolenaar848fadd2022-01-30 15:28:30 +00002519 RefStr = g:FuncNoArgRetString
Bram Moolenaarec5929d2020-04-07 20:53:39 +02002520 RefStr = FuncOneArgRetString
Bram Moolenaar62aec932022-01-29 21:45:34 +00002521 v9.CheckDefFailure(['var RefStr: func: string', 'RefStr = g:FuncNoArgNoRet'], 'E1012: Type mismatch; expected func(...): string but got func()')
2522 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 +02002523enddef
2524
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002525def Test_func_type_fails()
Bram Moolenaar62aec932022-01-29 21:45:34 +00002526 v9.CheckDefFailure(['var ref1: func()'], 'E704:')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002527
Bram Moolenaar62aec932022-01-29 21:45:34 +00002528 v9.CheckDefFailure(['var Ref1: func()', 'Ref1 = g:FuncNoArgRetNumber'], 'E1012: Type mismatch; expected func() but got func(): number')
2529 v9.CheckDefFailure(['var Ref1: func()', 'Ref1 = g:FuncOneArgNoRet'], 'E1012: Type mismatch; expected func() but got func(number)')
Bram Moolenaar848fadd2022-01-30 15:28:30 +00002530 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 +00002531 v9.CheckDefFailure(['var Ref1: func(bool)', 'Ref1 = g:FuncTwoArgNoRet'], 'E1012: Type mismatch; expected func(bool) but got func(bool, number)')
2532 v9.CheckDefFailure(['var Ref1: func(?bool)', 'Ref1 = g:FuncTwoArgNoRet'], 'E1012: Type mismatch; expected func(?bool) but got func(bool, number)')
2533 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 +02002534
Bram Moolenaar62aec932022-01-29 21:45:34 +00002535 v9.CheckDefFailure(['var RefWrong: func(string ,number)'], 'E1068:')
2536 v9.CheckDefFailure(['var RefWrong: func(string,number)'], 'E1069:')
2537 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:')
2538 v9.CheckDefFailure(['var RefWrong: func(bool):string'], 'E1069:')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002539enddef
2540
Bram Moolenaar89228602020-04-05 22:14:54 +02002541def Test_func_return_type()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002542 var nr: number
Bram Moolenaar62aec932022-01-29 21:45:34 +00002543 nr = g:FuncNoArgRetNumber()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002544 nr->assert_equal(1234)
Bram Moolenaar89228602020-04-05 22:14:54 +02002545
2546 nr = FuncOneArgRetAny(122)
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002547 nr->assert_equal(122)
Bram Moolenaar89228602020-04-05 22:14:54 +02002548
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002549 var str: string
Bram Moolenaar89228602020-04-05 22:14:54 +02002550 str = FuncOneArgRetAny('yes')
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002551 str->assert_equal('yes')
Bram Moolenaar89228602020-04-05 22:14:54 +02002552
Bram Moolenaar62aec932022-01-29 21:45:34 +00002553 v9.CheckDefFailure(['var str: string', 'str = g:FuncNoArgRetNumber()'], 'E1012: Type mismatch; expected string but got number')
Bram Moolenaar89228602020-04-05 22:14:54 +02002554enddef
2555
Bram Moolenaar6abd3dc2020-10-04 14:17:32 +02002556def Test_func_common_type()
2557 def FuncOne(n: number): number
2558 return n
2559 enddef
2560 def FuncTwo(s: string): number
2561 return len(s)
2562 enddef
2563 def FuncThree(n: number, s: string): number
2564 return n + len(s)
2565 enddef
2566 var list = [FuncOne, FuncTwo, FuncThree]
2567 assert_equal(8, list[0](8))
2568 assert_equal(4, list[1]('word'))
2569 assert_equal(7, list[2](3, 'word'))
2570enddef
2571
Bram Moolenaar62aec932022-01-29 21:45:34 +00002572def s:MultiLine(
Bram Moolenaar5e774c72020-04-12 21:53:00 +02002573 arg1: string,
2574 arg2 = 1234,
2575 ...rest: list<string>
2576 ): string
2577 return arg1 .. arg2 .. join(rest, '-')
2578enddef
2579
Bram Moolenaar2c330432020-04-13 14:41:35 +02002580def MultiLineComment(
2581 arg1: string, # comment
2582 arg2 = 1234, # comment
2583 ...rest: list<string> # comment
2584 ): string # comment
2585 return arg1 .. arg2 .. join(rest, '-')
2586enddef
2587
Bram Moolenaar5e774c72020-04-12 21:53:00 +02002588def Test_multiline()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002589 MultiLine('text')->assert_equal('text1234')
2590 MultiLine('text', 777)->assert_equal('text777')
2591 MultiLine('text', 777, 'one')->assert_equal('text777one')
2592 MultiLine('text', 777, 'one', 'two')->assert_equal('text777one-two')
Bram Moolenaar5e774c72020-04-12 21:53:00 +02002593enddef
2594
Bram Moolenaar23e03252020-04-12 22:22:31 +02002595func Test_multiline_not_vim9()
Bram Moolenaar62aec932022-01-29 21:45:34 +00002596 call s:MultiLine('text')->assert_equal('text1234')
2597 call s:MultiLine('text', 777)->assert_equal('text777')
2598 call s:MultiLine('text', 777, 'one')->assert_equal('text777one')
2599 call s:MultiLine('text', 777, 'one', 'two')->assert_equal('text777one-two')
Bram Moolenaar23e03252020-04-12 22:22:31 +02002600endfunc
2601
Bram Moolenaar5e774c72020-04-12 21:53:00 +02002602
Bram Moolenaaree4e0c12020-04-06 21:35:05 +02002603" When using CheckScriptFailure() for the below test, E1010 is generated instead
2604" of E1056.
2605func Test_E1056_1059()
2606 let caught_1056 = 0
2607 try
2608 def F():
2609 return 1
2610 enddef
2611 catch /E1056:/
2612 let caught_1056 = 1
2613 endtry
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002614 eval caught_1056->assert_equal(1)
Bram Moolenaaree4e0c12020-04-06 21:35:05 +02002615
2616 let caught_1059 = 0
2617 try
2618 def F5(items : list)
2619 echo 'a'
2620 enddef
2621 catch /E1059:/
2622 let caught_1059 = 1
2623 endtry
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002624 eval caught_1059->assert_equal(1)
Bram Moolenaaree4e0c12020-04-06 21:35:05 +02002625endfunc
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002626
Bram Moolenaar015f4262020-05-05 21:25:22 +02002627func DelMe()
2628 echo 'DelMe'
2629endfunc
2630
Bram Moolenaarbf8feb52020-08-08 14:26:31 +02002631def Test_error_reporting()
2632 # comment lines at the start of the function
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002633 var lines =<< trim END
Bram Moolenaarbf8feb52020-08-08 14:26:31 +02002634 " comment
2635 def Func()
2636 # comment
2637 # comment
2638 invalid
2639 enddef
2640 defcompile
2641 END
Bram Moolenaar08052222020-09-14 17:04:31 +02002642 writefile(lines, 'Xdef')
Bram Moolenaarbf8feb52020-08-08 14:26:31 +02002643 try
2644 source Xdef
Bram Moolenaar7517ffd2020-08-14 18:35:07 +02002645 assert_report('should have failed')
Bram Moolenaarbf8feb52020-08-08 14:26:31 +02002646 catch /E476:/
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002647 v:exception->assert_match('Invalid command: invalid')
2648 v:throwpoint->assert_match(', line 3$')
Bram Moolenaarbf8feb52020-08-08 14:26:31 +02002649 endtry
Bram Moolenaar2d870f82020-12-05 13:41:01 +01002650 delfunc! g:Func
Bram Moolenaarbf8feb52020-08-08 14:26:31 +02002651
2652 # comment lines after the start of the function
2653 lines =<< trim END
2654 " comment
2655 def Func()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002656 var x = 1234
Bram Moolenaarbf8feb52020-08-08 14:26:31 +02002657 # comment
2658 # comment
2659 invalid
2660 enddef
2661 defcompile
2662 END
Bram Moolenaar08052222020-09-14 17:04:31 +02002663 writefile(lines, 'Xdef')
Bram Moolenaarbf8feb52020-08-08 14:26:31 +02002664 try
2665 source Xdef
Bram Moolenaar7517ffd2020-08-14 18:35:07 +02002666 assert_report('should have failed')
Bram Moolenaarbf8feb52020-08-08 14:26:31 +02002667 catch /E476:/
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002668 v:exception->assert_match('Invalid command: invalid')
2669 v:throwpoint->assert_match(', line 4$')
Bram Moolenaarbf8feb52020-08-08 14:26:31 +02002670 endtry
Bram Moolenaar2d870f82020-12-05 13:41:01 +01002671 delfunc! g:Func
Bram Moolenaarbf8feb52020-08-08 14:26:31 +02002672
Bram Moolenaar7517ffd2020-08-14 18:35:07 +02002673 lines =<< trim END
2674 vim9script
2675 def Func()
Bram Moolenaare0de1712020-12-02 17:36:54 +01002676 var db = {foo: 1, bar: 2}
Bram Moolenaar7517ffd2020-08-14 18:35:07 +02002677 # comment
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002678 var x = db.asdf
Bram Moolenaar7517ffd2020-08-14 18:35:07 +02002679 enddef
2680 defcompile
2681 Func()
2682 END
Bram Moolenaar08052222020-09-14 17:04:31 +02002683 writefile(lines, 'Xdef')
Bram Moolenaar7517ffd2020-08-14 18:35:07 +02002684 try
2685 source Xdef
2686 assert_report('should have failed')
2687 catch /E716:/
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002688 v:throwpoint->assert_match('_Func, line 3$')
Bram Moolenaar7517ffd2020-08-14 18:35:07 +02002689 endtry
Bram Moolenaar2d870f82020-12-05 13:41:01 +01002690 delfunc! g:Func
Bram Moolenaar7517ffd2020-08-14 18:35:07 +02002691
Bram Moolenaar08052222020-09-14 17:04:31 +02002692 delete('Xdef')
Bram Moolenaarbf8feb52020-08-08 14:26:31 +02002693enddef
2694
Bram Moolenaar015f4262020-05-05 21:25:22 +02002695def Test_deleted_function()
Bram Moolenaar62aec932022-01-29 21:45:34 +00002696 v9.CheckDefExecFailure([
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002697 'var RefMe: func = function("g:DelMe")',
Bram Moolenaar015f4262020-05-05 21:25:22 +02002698 'delfunc g:DelMe',
2699 'echo RefMe()'], 'E117:')
2700enddef
2701
2702def Test_unknown_function()
Bram Moolenaar62aec932022-01-29 21:45:34 +00002703 v9.CheckDefExecFailure([
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002704 'var Ref: func = function("NotExist")',
Bram Moolenaar9b7bf9e2020-07-11 22:14:59 +02002705 'delfunc g:NotExist'], 'E700:')
Bram Moolenaar015f4262020-05-05 21:25:22 +02002706enddef
2707
Bram Moolenaar62aec932022-01-29 21:45:34 +00002708def s:RefFunc(Ref: func(any): any): string
Bram Moolenaarc8cd2b32020-05-01 19:29:08 +02002709 return Ref('more')
2710enddef
2711
2712def Test_closure_simple()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002713 var local = 'some '
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01002714 RefFunc((s) => local .. s)->assert_equal('some more')
Bram Moolenaarc8cd2b32020-05-01 19:29:08 +02002715enddef
2716
Bram Moolenaar62aec932022-01-29 21:45:34 +00002717def s:MakeRef()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002718 var local = 'some '
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01002719 g:Ref = (s) => local .. s
Bram Moolenaarbf67ea12020-05-02 17:52:42 +02002720enddef
2721
2722def Test_closure_ref_after_return()
2723 MakeRef()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002724 g:Ref('thing')->assert_equal('some thing')
Bram Moolenaarbf67ea12020-05-02 17:52:42 +02002725 unlet g:Ref
2726enddef
2727
Bram Moolenaar62aec932022-01-29 21:45:34 +00002728def s:MakeTwoRefs()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002729 var local = ['some']
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01002730 g:Extend = (s) => local->add(s)
2731 g:Read = () => local
Bram Moolenaar5adc55c2020-05-02 23:12:58 +02002732enddef
2733
2734def Test_closure_two_refs()
2735 MakeTwoRefs()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002736 join(g:Read(), ' ')->assert_equal('some')
Bram Moolenaar5adc55c2020-05-02 23:12:58 +02002737 g:Extend('more')
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002738 join(g:Read(), ' ')->assert_equal('some more')
Bram Moolenaar5adc55c2020-05-02 23:12:58 +02002739 g:Extend('even')
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002740 join(g:Read(), ' ')->assert_equal('some more even')
Bram Moolenaar5adc55c2020-05-02 23:12:58 +02002741
2742 unlet g:Extend
2743 unlet g:Read
2744enddef
2745
Bram Moolenaar62aec932022-01-29 21:45:34 +00002746def s:ReadRef(Ref: func(): list<string>): string
Bram Moolenaar5adc55c2020-05-02 23:12:58 +02002747 return join(Ref(), ' ')
2748enddef
2749
Bram Moolenaar62aec932022-01-29 21:45:34 +00002750def s:ExtendRef(Ref: func(string): list<string>, add: string)
Bram Moolenaar5adc55c2020-05-02 23:12:58 +02002751 Ref(add)
2752enddef
2753
2754def Test_closure_two_indirect_refs()
Bram Moolenaarf7779c62020-05-03 15:38:16 +02002755 MakeTwoRefs()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002756 ReadRef(g:Read)->assert_equal('some')
Bram Moolenaar5adc55c2020-05-02 23:12:58 +02002757 ExtendRef(g:Extend, 'more')
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002758 ReadRef(g:Read)->assert_equal('some more')
Bram Moolenaar5adc55c2020-05-02 23:12:58 +02002759 ExtendRef(g:Extend, 'even')
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002760 ReadRef(g:Read)->assert_equal('some more even')
Bram Moolenaar5adc55c2020-05-02 23:12:58 +02002761
2762 unlet g:Extend
2763 unlet g:Read
2764enddef
Bram Moolenaarbf67ea12020-05-02 17:52:42 +02002765
Bram Moolenaar62aec932022-01-29 21:45:34 +00002766def s:MakeArgRefs(theArg: string)
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002767 var local = 'loc_val'
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01002768 g:UseArg = (s) => theArg .. '/' .. local .. '/' .. s
Bram Moolenaar2fd4cd72020-05-03 22:30:49 +02002769enddef
2770
Bram Moolenaar62aec932022-01-29 21:45:34 +00002771def s:MakeArgRefsVarargs(theArg: string, ...rest: list<string>)
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002772 var local = 'the_loc'
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01002773 g:UseVararg = (s) => theArg .. '/' .. local .. '/' .. s .. '/' .. join(rest)
Bram Moolenaar2fd4cd72020-05-03 22:30:49 +02002774enddef
2775
2776def Test_closure_using_argument()
2777 MakeArgRefs('arg_val')
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002778 g:UseArg('call_val')->assert_equal('arg_val/loc_val/call_val')
Bram Moolenaar2fd4cd72020-05-03 22:30:49 +02002779
2780 MakeArgRefsVarargs('arg_val', 'one', 'two')
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002781 g:UseVararg('call_val')->assert_equal('arg_val/the_loc/call_val/one two')
Bram Moolenaar2fd4cd72020-05-03 22:30:49 +02002782
2783 unlet g:UseArg
2784 unlet g:UseVararg
Bram Moolenaar44ec21c2021-02-12 21:50:57 +01002785
2786 var lines =<< trim END
2787 vim9script
2788 def Test(Fun: func(number): number): list<number>
2789 return map([1, 2, 3], (_, i) => Fun(i))
2790 enddef
2791 def Inc(nr: number): number
2792 return nr + 2
2793 enddef
2794 assert_equal([3, 4, 5], Test(Inc))
2795 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002796 v9.CheckScriptSuccess(lines)
Bram Moolenaar2fd4cd72020-05-03 22:30:49 +02002797enddef
2798
Bram Moolenaar62aec932022-01-29 21:45:34 +00002799def s:MakeGetAndAppendRefs()
Bram Moolenaar85d5e2b2020-10-10 14:13:01 +02002800 var local = 'a'
2801
2802 def Append(arg: string)
2803 local ..= arg
2804 enddef
2805 g:Append = Append
2806
2807 def Get(): string
2808 return local
2809 enddef
2810 g:Get = Get
2811enddef
2812
2813def Test_closure_append_get()
2814 MakeGetAndAppendRefs()
2815 g:Get()->assert_equal('a')
2816 g:Append('-b')
2817 g:Get()->assert_equal('a-b')
2818 g:Append('-c')
2819 g:Get()->assert_equal('a-b-c')
2820
2821 unlet g:Append
2822 unlet g:Get
2823enddef
Bram Moolenaarb68b3462020-05-06 21:06:30 +02002824
Bram Moolenaar04b12692020-05-04 23:24:44 +02002825def Test_nested_closure()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002826 var local = 'text'
Bram Moolenaar04b12692020-05-04 23:24:44 +02002827 def Closure(arg: string): string
2828 return local .. arg
2829 enddef
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002830 Closure('!!!')->assert_equal('text!!!')
Bram Moolenaar04b12692020-05-04 23:24:44 +02002831enddef
2832
Bram Moolenaar62aec932022-01-29 21:45:34 +00002833func s:GetResult(Ref)
Bram Moolenaar6f5b6df2020-05-16 21:20:12 +02002834 return a:Ref('some')
2835endfunc
2836
2837def Test_call_closure_not_compiled()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002838 var text = 'text'
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01002839 g:Ref = (s) => s .. text
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002840 GetResult(g:Ref)->assert_equal('sometext')
Bram Moolenaar6f5b6df2020-05-16 21:20:12 +02002841enddef
2842
Bram Moolenaar7cbfaa52020-09-18 21:25:32 +02002843def Test_double_closure_fails()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002844 var lines =<< trim END
Bram Moolenaar7cbfaa52020-09-18 21:25:32 +02002845 vim9script
2846 def Func()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002847 var name = 0
2848 for i in range(2)
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01002849 timer_start(0, () => name)
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002850 endfor
Bram Moolenaar7cbfaa52020-09-18 21:25:32 +02002851 enddef
2852 Func()
2853 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002854 v9.CheckScriptSuccess(lines)
Bram Moolenaar7cbfaa52020-09-18 21:25:32 +02002855enddef
2856
Bram Moolenaar85d5e2b2020-10-10 14:13:01 +02002857def Test_nested_closure_used()
2858 var lines =<< trim END
2859 vim9script
2860 def Func()
2861 var x = 'hello'
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01002862 var Closure = () => x
2863 g:Myclosure = () => Closure()
Bram Moolenaar85d5e2b2020-10-10 14:13:01 +02002864 enddef
2865 Func()
2866 assert_equal('hello', g:Myclosure())
2867 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002868 v9.CheckScriptSuccess(lines)
Bram Moolenaar85d5e2b2020-10-10 14:13:01 +02002869enddef
Bram Moolenaar0876c782020-10-07 19:08:04 +02002870
Bram Moolenaarc70bdab2020-09-26 19:59:38 +02002871def Test_nested_closure_fails()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002872 var lines =<< trim END
Bram Moolenaarc70bdab2020-09-26 19:59:38 +02002873 vim9script
2874 def FuncA()
2875 FuncB(0)
2876 enddef
2877 def FuncB(n: number): list<string>
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01002878 return map([0], (_, v) => n)
Bram Moolenaarc70bdab2020-09-26 19:59:38 +02002879 enddef
2880 FuncA()
2881 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002882 v9.CheckScriptFailure(lines, 'E1012:')
Bram Moolenaarc70bdab2020-09-26 19:59:38 +02002883enddef
2884
Bram Moolenaarf112f302020-12-20 17:47:52 +01002885def Test_global_closure()
2886 var lines =<< trim END
2887 vim9script
2888 def ReverseEveryNLines(n: number, line1: number, line2: number)
2889 var mods = 'sil keepj keepp lockm '
2890 var range = ':' .. line1 .. ',' .. line2
2891 def g:Offset(): number
2892 var offset = (line('.') - line1 + 1) % n
2893 return offset != 0 ? offset : n
2894 enddef
2895 exe mods .. range .. 'g/^/exe "m .-" .. g:Offset()'
2896 enddef
2897
2898 new
2899 repeat(['aaa', 'bbb', 'ccc'], 3)->setline(1)
2900 ReverseEveryNLines(3, 1, 9)
2901 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002902 v9.CheckScriptSuccess(lines)
Bram Moolenaarf112f302020-12-20 17:47:52 +01002903 var expected = repeat(['ccc', 'bbb', 'aaa'], 3)
2904 assert_equal(expected, getline(1, 9))
2905 bwipe!
2906enddef
2907
Bram Moolenaarcd45ed02020-12-22 17:35:54 +01002908def Test_global_closure_called_directly()
2909 var lines =<< trim END
2910 vim9script
2911 def Outer()
2912 var x = 1
2913 def g:Inner()
2914 var y = x
2915 x += 1
2916 assert_equal(1, y)
2917 enddef
2918 g:Inner()
2919 assert_equal(2, x)
2920 enddef
2921 Outer()
2922 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002923 v9.CheckScriptSuccess(lines)
Bram Moolenaarcd45ed02020-12-22 17:35:54 +01002924 delfunc g:Inner
2925enddef
2926
Bram Moolenaar69c76172021-12-02 16:38:52 +00002927def Test_closure_called_from_legacy()
2928 var lines =<< trim END
2929 vim9script
2930 def Func()
2931 var outer = 'foo'
2932 var F = () => {
2933 outer = 'bar'
2934 }
2935 execute printf('call %s()', string(F))
2936 enddef
2937 Func()
2938 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002939 v9.CheckScriptFailure(lines, 'E1248')
Bram Moolenaar69c76172021-12-02 16:38:52 +00002940enddef
2941
Bram Moolenaar34c54eb2020-11-25 19:15:19 +01002942def Test_failure_in_called_function()
2943 # this was using the frame index as the return value
2944 var lines =<< trim END
2945 vim9script
2946 au TerminalWinOpen * eval [][0]
2947 def PopupTerm(a: any)
2948 # make sure typvals on stack are string
2949 ['a', 'b', 'c', 'd', 'e', 'f', 'g']->join()
2950 FireEvent()
2951 enddef
2952 def FireEvent()
2953 do TerminalWinOpen
2954 enddef
2955 # use try/catch to make eval fail
2956 try
2957 call PopupTerm(0)
2958 catch
2959 endtry
2960 au! TerminalWinOpen
2961 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002962 v9.CheckScriptSuccess(lines)
Bram Moolenaar34c54eb2020-11-25 19:15:19 +01002963enddef
2964
Bram Moolenaar5366e1a2020-10-01 13:01:34 +02002965def Test_nested_lambda()
2966 var lines =<< trim END
2967 vim9script
2968 def Func()
2969 var x = 4
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01002970 var Lambda1 = () => 7
2971 var Lambda2 = () => [Lambda1(), x]
Bram Moolenaar5366e1a2020-10-01 13:01:34 +02002972 var res = Lambda2()
2973 assert_equal([7, 4], res)
2974 enddef
2975 Func()
2976 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002977 v9.CheckScriptSuccess(lines)
Bram Moolenaar5366e1a2020-10-01 13:01:34 +02002978enddef
2979
Bram Moolenaarc04f2a42021-06-09 19:30:03 +02002980def Test_double_nested_lambda()
2981 var lines =<< trim END
2982 vim9script
2983 def F(head: string): func(string): func(string): string
2984 return (sep: string): func(string): string => ((tail: string): string => {
2985 return head .. sep .. tail
2986 })
2987 enddef
2988 assert_equal('hello-there', F('hello')('-')('there'))
2989 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002990 v9.CheckScriptSuccess(lines)
Bram Moolenaarc04f2a42021-06-09 19:30:03 +02002991enddef
2992
Bram Moolenaar074f84c2021-05-18 11:47:44 +02002993def Test_nested_inline_lambda()
Bram Moolenaar074f84c2021-05-18 11:47:44 +02002994 var lines =<< trim END
2995 vim9script
2996 def F(text: string): func(string): func(string): string
2997 return (arg: string): func(string): string => ((sep: string): string => {
Bram Moolenaar23e2e112021-08-03 21:16:18 +02002998 return sep .. arg .. text
Bram Moolenaar074f84c2021-05-18 11:47:44 +02002999 })
3000 enddef
Bram Moolenaar23e2e112021-08-03 21:16:18 +02003001 assert_equal('--there++', F('++')('there')('--'))
Bram Moolenaar074f84c2021-05-18 11:47:44 +02003002 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003003 v9.CheckScriptSuccess(lines)
Bram Moolenaar5245beb2021-07-15 22:03:50 +02003004
3005 lines =<< trim END
3006 vim9script
3007 echo range(4)->mapnew((_, v) => {
3008 return range(v) ->mapnew((_, s) => {
3009 return string(s)
3010 })
3011 })
3012 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003013 v9.CheckScriptSuccess(lines)
Bram Moolenaarc6ba2f92021-07-18 13:42:29 +02003014
3015 lines =<< trim END
3016 vim9script
3017
Bram Moolenaara749a422022-02-12 19:52:25 +00003018 def Func()
Bram Moolenaarc6ba2f92021-07-18 13:42:29 +02003019 range(10)
3020 ->mapnew((_, _) => ({
3021 key: range(10)->mapnew((_, _) => {
3022 return ' '
3023 }),
3024 }))
3025 enddef
3026
3027 defcomp
3028 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003029 v9.CheckScriptSuccess(lines)
Bram Moolenaar074f84c2021-05-18 11:47:44 +02003030enddef
3031
Bram Moolenaar52bf81c2020-11-17 18:50:44 +01003032def Shadowed(): list<number>
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01003033 var FuncList: list<func: number> = [() => 42]
Bram Moolenaar75ab91f2021-01-10 22:42:50 +01003034 return FuncList->mapnew((_, Shadowed) => Shadowed())
Bram Moolenaar52bf81c2020-11-17 18:50:44 +01003035enddef
3036
3037def Test_lambda_arg_shadows_func()
Bram Moolenaar62aec932022-01-29 21:45:34 +00003038 assert_equal([42], g:Shadowed())
Bram Moolenaar52bf81c2020-11-17 18:50:44 +01003039enddef
3040
Bram Moolenaar21dc8f12022-03-16 17:54:17 +00003041def Test_compiling_referenced_func_no_shadow()
3042 var lines =<< trim END
3043 vim9script
3044
3045 def InitializeReply(lspserver: dict<any>)
3046 enddef
3047
3048 def ProcessReply(lspserver: dict<any>)
3049 var lsp_reply_handlers: dict<func> =
3050 { 'initialize': InitializeReply }
3051 lsp_reply_handlers['initialize'](lspserver)
3052 enddef
3053
3054 call ProcessReply({})
3055 END
3056 v9.CheckScriptSuccess(lines)
3057enddef
3058
Bram Moolenaar62aec932022-01-29 21:45:34 +00003059def s:Line_continuation_in_def(dir: string = ''): string
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02003060 var path: string = empty(dir)
3061 \ ? 'empty'
3062 \ : 'full'
3063 return path
Bram Moolenaaracd4c5e2020-06-22 19:39:03 +02003064enddef
3065
3066def Test_line_continuation_in_def()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02003067 Line_continuation_in_def('.')->assert_equal('full')
Bram Moolenaaracd4c5e2020-06-22 19:39:03 +02003068enddef
3069
Bram Moolenaar2ea95b62020-11-19 21:47:56 +01003070def Test_script_var_in_lambda()
3071 var lines =<< trim END
3072 vim9script
3073 var script = 'test'
Bram Moolenaarbb8a7ce2021-04-10 20:10:26 +02003074 assert_equal(['test'], map(['one'], (_, _) => script))
Bram Moolenaar2ea95b62020-11-19 21:47:56 +01003075 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003076 v9.CheckScriptSuccess(lines)
Bram Moolenaar2ea95b62020-11-19 21:47:56 +01003077enddef
3078
Bram Moolenaar62aec932022-01-29 21:45:34 +00003079def s:Line_continuation_in_lambda(): list<string>
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02003080 var x = range(97, 100)
Bram Moolenaar75ab91f2021-01-10 22:42:50 +01003081 ->mapnew((_, v) => nr2char(v)
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01003082 ->toupper())
Bram Moolenaar7a4b8982020-07-08 17:36:21 +02003083 ->reverse()
3084 return x
3085enddef
3086
3087def Test_line_continuation_in_lambda()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02003088 Line_continuation_in_lambda()->assert_equal(['D', 'C', 'B', 'A'])
Bram Moolenaarf898f7c2021-01-16 18:09:52 +01003089
3090 var lines =<< trim END
3091 vim9script
3092 var res = [{n: 1, m: 2, s: 'xxx'}]
3093 ->mapnew((_, v: dict<any>): string => printf('%d:%d:%s',
3094 v.n,
3095 v.m,
3096 substitute(v.s, '.*', 'yyy', '')
3097 ))
3098 assert_equal(['1:2:yyy'], res)
3099 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003100 v9.CheckScriptSuccess(lines)
Bram Moolenaar7a4b8982020-07-08 17:36:21 +02003101enddef
3102
Bram Moolenaarb6571982021-01-08 22:24:19 +01003103def Test_list_lambda()
3104 timer_start(1000, (_) => 0)
3105 var body = execute(timer_info()[0].callback
3106 ->string()
3107 ->substitute("('", ' ', '')
3108 ->substitute("')", '', '')
3109 ->substitute('function\zs', ' ', ''))
Bram Moolenaar767034c2021-04-09 17:24:52 +02003110 assert_match('def <lambda>\d\+(_: any): number\n1 return 0\n enddef', body)
Bram Moolenaarb6571982021-01-08 22:24:19 +01003111enddef
3112
Bram Moolenaar3c77b6a2021-07-25 18:07:00 +02003113def Test_lambda_block_variable()
Bram Moolenaar88421d62021-07-24 14:14:52 +02003114 var lines =<< trim END
3115 vim9script
3116 var flist: list<func>
3117 for i in range(10)
3118 var inloop = i
3119 flist[i] = () => inloop
3120 endfor
3121 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003122 v9.CheckScriptSuccess(lines)
Bram Moolenaar88421d62021-07-24 14:14:52 +02003123
3124 lines =<< trim END
3125 vim9script
3126 if true
3127 var outloop = 5
3128 var flist: list<func>
3129 for i in range(10)
3130 flist[i] = () => outloop
3131 endfor
3132 endif
3133 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003134 v9.CheckScriptSuccess(lines)
Bram Moolenaar88421d62021-07-24 14:14:52 +02003135
3136 lines =<< trim END
3137 vim9script
3138 if true
3139 var outloop = 5
3140 endif
3141 var flist: list<func>
3142 for i in range(10)
3143 flist[i] = () => outloop
3144 endfor
3145 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003146 v9.CheckScriptFailure(lines, 'E1001: Variable not found: outloop', 1)
Bram Moolenaar3c77b6a2021-07-25 18:07:00 +02003147
3148 lines =<< trim END
3149 vim9script
3150 for i in range(10)
3151 var Ref = () => 0
3152 endfor
3153 assert_equal(0, ((i) => 0)(0))
3154 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003155 v9.CheckScriptSuccess(lines)
Bram Moolenaar88421d62021-07-24 14:14:52 +02003156enddef
3157
Bram Moolenaar96cf4ba2021-04-24 14:15:41 +02003158def Test_legacy_lambda()
3159 legacy echo {x -> 'hello ' .. x}('foo')
Bram Moolenaardc4c2302021-04-25 13:54:42 +02003160
Bram Moolenaar96cf4ba2021-04-24 14:15:41 +02003161 var lines =<< trim END
3162 echo {x -> 'hello ' .. x}('foo')
3163 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003164 v9.CheckDefAndScriptFailure(lines, 'E720:')
Bram Moolenaardc4c2302021-04-25 13:54:42 +02003165
3166 lines =<< trim END
3167 vim9script
3168 def Func()
3169 echo (() => 'no error')()
3170 enddef
3171 legacy call s:Func()
3172 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003173 v9.CheckScriptSuccess(lines)
Bram Moolenaar96cf4ba2021-04-24 14:15:41 +02003174enddef
3175
Bram Moolenaarce024c32021-06-26 13:00:49 +02003176def Test_legacy()
3177 var lines =<< trim END
3178 vim9script
3179 func g:LegacyFunction()
3180 let g:legacyvar = 1
3181 endfunc
3182 def Testit()
3183 legacy call g:LegacyFunction()
3184 enddef
3185 Testit()
3186 assert_equal(1, g:legacyvar)
3187 unlet g:legacyvar
3188 delfunc g:LegacyFunction
3189 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003190 v9.CheckScriptSuccess(lines)
Bram Moolenaarce024c32021-06-26 13:00:49 +02003191enddef
3192
Bram Moolenaarc3cb1c92021-06-02 16:47:53 +02003193def Test_legacy_errors()
3194 for cmd in ['if', 'elseif', 'else', 'endif',
3195 'for', 'endfor', 'continue', 'break',
3196 'while', 'endwhile',
3197 'try', 'catch', 'finally', 'endtry']
Bram Moolenaar62aec932022-01-29 21:45:34 +00003198 v9.CheckDefFailure(['legacy ' .. cmd .. ' expr'], 'E1189:')
Bram Moolenaarc3cb1c92021-06-02 16:47:53 +02003199 endfor
3200enddef
3201
Bram Moolenaarb1b6f4d2021-09-13 18:25:54 +02003202def Test_call_legacy_with_dict()
3203 var lines =<< trim END
3204 vim9script
3205 func Legacy() dict
3206 let g:result = self.value
3207 endfunc
3208 def TestDirect()
3209 var d = {value: 'yes', func: Legacy}
3210 d.func()
3211 enddef
3212 TestDirect()
3213 assert_equal('yes', g:result)
3214 unlet g:result
3215
3216 def TestIndirect()
3217 var d = {value: 'foo', func: Legacy}
3218 var Fi = d.func
3219 Fi()
3220 enddef
3221 TestIndirect()
3222 assert_equal('foo', g:result)
3223 unlet g:result
3224
3225 var d = {value: 'bar', func: Legacy}
3226 d.func()
3227 assert_equal('bar', g:result)
3228 unlet g:result
3229 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003230 v9.CheckScriptSuccess(lines)
Bram Moolenaarb1b6f4d2021-09-13 18:25:54 +02003231enddef
3232
Bram Moolenaar62aec932022-01-29 21:45:34 +00003233def s:DoFilterThis(a: string): list<string>
Bram Moolenaarab360522021-01-10 14:02:28 +01003234 # closure nested inside another closure using argument
3235 var Filter = (l) => filter(l, (_, v) => stridx(v, a) == 0)
3236 return ['x', 'y', 'a', 'x2', 'c']->Filter()
3237enddef
3238
3239def Test_nested_closure_using_argument()
3240 assert_equal(['x', 'x2'], DoFilterThis('x'))
3241enddef
3242
Bram Moolenaar0186e582021-01-10 18:33:11 +01003243def Test_triple_nested_closure()
3244 var what = 'x'
3245 var Match = (val: string, cmp: string): bool => stridx(val, cmp) == 0
3246 var Filter = (l) => filter(l, (_, v) => Match(v, what))
3247 assert_equal(['x', 'x2'], ['x', 'y', 'a', 'x2', 'c']->Filter())
3248enddef
3249
Bram Moolenaar8f510af2020-07-05 18:48:23 +02003250func Test_silent_echo()
Bram Moolenaar47e7d702020-07-05 18:18:42 +02003251 CheckScreendump
Bram Moolenaar3b309f12021-12-13 18:19:55 +00003252 call Run_Test_silent_echo()
3253endfunc
Bram Moolenaar47e7d702020-07-05 18:18:42 +02003254
Bram Moolenaar3b309f12021-12-13 18:19:55 +00003255def Run_Test_silent_echo()
3256 var lines =<< trim END
Bram Moolenaar47e7d702020-07-05 18:18:42 +02003257 vim9script
3258 def EchoNothing()
3259 silent echo ''
3260 enddef
3261 defcompile
3262 END
Bram Moolenaar3b309f12021-12-13 18:19:55 +00003263 writefile(lines, 'XTest_silent_echo')
Bram Moolenaar47e7d702020-07-05 18:18:42 +02003264
Bram Moolenaar3b309f12021-12-13 18:19:55 +00003265 # Check that the balloon shows up after a mouse move
Bram Moolenaar62aec932022-01-29 21:45:34 +00003266 var buf = g:RunVimInTerminal('-S XTest_silent_echo', {'rows': 6})
Bram Moolenaar3b309f12021-12-13 18:19:55 +00003267 term_sendkeys(buf, ":abc")
Bram Moolenaar62aec932022-01-29 21:45:34 +00003268 g:VerifyScreenDump(buf, 'Test_vim9_silent_echo', {})
Bram Moolenaar47e7d702020-07-05 18:18:42 +02003269
Bram Moolenaar3b309f12021-12-13 18:19:55 +00003270 # clean up
Bram Moolenaar62aec932022-01-29 21:45:34 +00003271 g:StopVimInTerminal(buf)
Bram Moolenaar3b309f12021-12-13 18:19:55 +00003272 delete('XTest_silent_echo')
3273enddef
Bram Moolenaar47e7d702020-07-05 18:18:42 +02003274
Bram Moolenaar171fb922020-10-28 16:54:47 +01003275def SilentlyError()
3276 execute('silent! invalid')
3277 g:did_it = 'yes'
3278enddef
3279
Bram Moolenaar62aec932022-01-29 21:45:34 +00003280func s:UserError()
Bram Moolenaar28ee8922020-10-28 20:20:00 +01003281 silent! invalid
3282endfunc
3283
3284def SilentlyUserError()
3285 UserError()
3286 g:did_it = 'yes'
3287enddef
Bram Moolenaar171fb922020-10-28 16:54:47 +01003288
3289" This can't be a :def function, because the assert would not be reached.
Bram Moolenaar171fb922020-10-28 16:54:47 +01003290func Test_ignore_silent_error()
3291 let g:did_it = 'no'
3292 call SilentlyError()
3293 call assert_equal('yes', g:did_it)
3294
Bram Moolenaar28ee8922020-10-28 20:20:00 +01003295 let g:did_it = 'no'
3296 call SilentlyUserError()
3297 call assert_equal('yes', g:did_it)
Bram Moolenaar171fb922020-10-28 16:54:47 +01003298
3299 unlet g:did_it
3300endfunc
3301
Bram Moolenaarcd030c42020-10-30 21:49:40 +01003302def Test_ignore_silent_error_in_filter()
3303 var lines =<< trim END
3304 vim9script
3305 def Filter(winid: number, key: string): bool
3306 if key == 'o'
3307 silent! eval [][0]
3308 return true
3309 endif
3310 return popup_filter_menu(winid, key)
3311 enddef
3312
Bram Moolenaare0de1712020-12-02 17:36:54 +01003313 popup_create('popup', {filter: Filter})
Bram Moolenaarcd030c42020-10-30 21:49:40 +01003314 feedkeys("o\r", 'xnt')
3315 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003316 v9.CheckScriptSuccess(lines)
Bram Moolenaarcd030c42020-10-30 21:49:40 +01003317enddef
3318
Bram Moolenaar62aec932022-01-29 21:45:34 +00003319def s:Fibonacci(n: number): number
Bram Moolenaar4b9bd692020-09-05 21:57:53 +02003320 if n < 2
3321 return n
3322 else
3323 return Fibonacci(n - 1) + Fibonacci(n - 2)
3324 endif
3325enddef
3326
Bram Moolenaar985116a2020-07-12 17:31:09 +02003327def Test_recursive_call()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02003328 Fibonacci(20)->assert_equal(6765)
Bram Moolenaar985116a2020-07-12 17:31:09 +02003329enddef
3330
Bram Moolenaar62aec932022-01-29 21:45:34 +00003331def s:TreeWalk(dir: string): list<any>
Bram Moolenaar75ab91f2021-01-10 22:42:50 +01003332 return readdir(dir)->mapnew((_, val) =>
Bram Moolenaar08f7a412020-07-13 20:41:08 +02003333 fnamemodify(dir .. '/' .. val, ':p')->isdirectory()
Bram Moolenaar2bede172020-11-19 18:53:18 +01003334 ? {[val]: TreeWalk(dir .. '/' .. val)}
Bram Moolenaar08f7a412020-07-13 20:41:08 +02003335 : val
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01003336 )
Bram Moolenaar08f7a412020-07-13 20:41:08 +02003337enddef
3338
3339def Test_closure_in_map()
3340 mkdir('XclosureDir/tdir', 'p')
3341 writefile(['111'], 'XclosureDir/file1')
3342 writefile(['222'], 'XclosureDir/file2')
3343 writefile(['333'], 'XclosureDir/tdir/file3')
3344
Bram Moolenaare0de1712020-12-02 17:36:54 +01003345 TreeWalk('XclosureDir')->assert_equal(['file1', 'file2', {tdir: ['file3']}])
Bram Moolenaar08f7a412020-07-13 20:41:08 +02003346
3347 delete('XclosureDir', 'rf')
3348enddef
3349
Bram Moolenaar7b5d5442020-10-04 13:42:34 +02003350def Test_invalid_function_name()
3351 var lines =<< trim END
3352 vim9script
3353 def s: list<string>
3354 END
Bram Moolenaara749a422022-02-12 19:52:25 +00003355 v9.CheckScriptFailure(lines, 'E1268:')
Bram Moolenaar7b5d5442020-10-04 13:42:34 +02003356
3357 lines =<< trim END
3358 vim9script
3359 def g: list<string>
3360 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003361 v9.CheckScriptFailure(lines, 'E129:')
Bram Moolenaar7b5d5442020-10-04 13:42:34 +02003362
3363 lines =<< trim END
3364 vim9script
3365 def <SID>: list<string>
3366 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003367 v9.CheckScriptFailure(lines, 'E884:')
Bram Moolenaar7b5d5442020-10-04 13:42:34 +02003368
3369 lines =<< trim END
3370 vim9script
3371 def F list<string>
3372 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003373 v9.CheckScriptFailure(lines, 'E488:')
Bram Moolenaar7b5d5442020-10-04 13:42:34 +02003374enddef
3375
Bram Moolenaara90afb92020-07-15 22:38:56 +02003376def Test_partial_call()
Bram Moolenaarf78da4f2021-08-01 15:40:31 +02003377 var lines =<< trim END
3378 var Xsetlist: func
3379 Xsetlist = function('setloclist', [0])
3380 Xsetlist([], ' ', {title: 'test'})
3381 getloclist(0, {title: 1})->assert_equal({title: 'test'})
Bram Moolenaara90afb92020-07-15 22:38:56 +02003382
Bram Moolenaarf78da4f2021-08-01 15:40:31 +02003383 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('setqflist')
3388 Xsetlist([], ' ', {title: 'test'})
3389 getqflist({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 Moolenaar6abd3dc2020-10-04 14:17:32 +02003394
Bram Moolenaarf78da4f2021-08-01 15:40:31 +02003395 var Len: func: number = function('len', ['word'])
3396 assert_equal(4, Len())
3397
3398 var RepeatFunc = function('repeat', ['o'])
3399 assert_equal('ooooo', RepeatFunc(5))
3400 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003401 v9.CheckDefAndScriptSuccess(lines)
Bram Moolenaarc66f6452021-08-19 21:08:30 +02003402
3403 lines =<< trim END
3404 vim9script
3405 def Foo(Parser: any)
3406 enddef
3407 var Expr: func(dict<any>): dict<any>
3408 const Call = Foo(Expr)
3409 END
Bram Moolenaar8acb9cc2022-03-08 13:18:55 +00003410 v9.CheckScriptFailure(lines, 'E1031:')
Bram Moolenaara90afb92020-07-15 22:38:56 +02003411enddef
3412
Bram Moolenaarcd1cda22022-02-16 21:48:25 +00003413def Test_partial_double_nested()
3414 var idx = 123
3415 var Get = () => idx
3416 var Ref = function(Get, [])
3417 var RefRef = function(Ref, [])
3418 assert_equal(123, RefRef())
3419enddef
3420
Bram Moolenaar673bcb12022-03-08 16:52:24 +00003421def Test_partial_null_function()
3422 var lines =<< trim END
3423 var d: dict<func> = {f: null_function}
3424 var Ref = d.f
Bram Moolenaared0c62e2022-03-08 19:43:55 +00003425 assert_equal('func(...): unknown', typename(Ref))
Bram Moolenaar673bcb12022-03-08 16:52:24 +00003426 END
3427 v9.CheckDefAndScriptSuccess(lines)
3428enddef
3429
Bram Moolenaarfe1bfc92022-02-06 13:55:03 +00003430" Using "idx" from a legacy global function does not work.
3431" This caused a crash when called from legacy context.
3432func Test_partial_call_fails()
3433 let lines =<< trim END
3434 vim9script
3435
3436 var l = ['a', 'b', 'c']
3437 def Iter(container: any): any
3438 var idx = -1
3439 var obj = {state: container}
Bram Moolenaarf681cfb2022-02-07 20:30:57 +00003440 def g:NextItem__(self: dict<any>): any
Bram Moolenaarfe1bfc92022-02-06 13:55:03 +00003441 ++idx
3442 return self.state[idx]
3443 enddef
Bram Moolenaarf681cfb2022-02-07 20:30:57 +00003444 obj.__next__ = function('g:NextItem__', [obj])
Bram Moolenaarfe1bfc92022-02-06 13:55:03 +00003445 return obj
3446 enddef
3447
3448 var it = Iter(l)
3449 echo it.__next__()
3450 END
3451 call writefile(lines, 'XpartialCall')
3452 try
3453 source XpartialCall
3454 catch /E1248:/
3455 endtry
3456 call delete('XpartialCall')
3457endfunc
3458
Bram Moolenaar2dd0a2c2020-08-08 15:10:27 +02003459def Test_cmd_modifier()
3460 tab echo '0'
Bram Moolenaar62aec932022-01-29 21:45:34 +00003461 v9.CheckDefFailure(['5tab echo 3'], 'E16:')
Bram Moolenaar2dd0a2c2020-08-08 15:10:27 +02003462enddef
3463
3464def Test_restore_modifiers()
3465 # check that when compiling a :def function command modifiers are not messed
3466 # up.
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02003467 var lines =<< trim END
Bram Moolenaar2dd0a2c2020-08-08 15:10:27 +02003468 vim9script
3469 set eventignore=
3470 autocmd QuickFixCmdPost * copen
3471 def AutocmdsDisabled()
Bram Moolenaarc3235272021-07-10 19:42:03 +02003472 eval 1 + 2
Bram Moolenaar2dd0a2c2020-08-08 15:10:27 +02003473 enddef
3474 func Func()
3475 noautocmd call s:AutocmdsDisabled()
3476 let g:ei_after = &eventignore
3477 endfunc
3478 Func()
3479 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003480 v9.CheckScriptSuccess(lines)
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02003481 g:ei_after->assert_equal('')
Bram Moolenaar2dd0a2c2020-08-08 15:10:27 +02003482enddef
3483
Bram Moolenaardfa3d552020-09-10 22:05:08 +02003484def StackTop()
Bram Moolenaarc3235272021-07-10 19:42:03 +02003485 eval 1 + 2
3486 eval 2 + 3
Bram Moolenaardfa3d552020-09-10 22:05:08 +02003487 # call not on fourth line
Bram Moolenaar62aec932022-01-29 21:45:34 +00003488 g:StackBot()
Bram Moolenaardfa3d552020-09-10 22:05:08 +02003489enddef
3490
3491def StackBot()
3492 # throw an error
3493 eval [][0]
3494enddef
3495
3496def Test_callstack_def()
3497 try
Bram Moolenaar62aec932022-01-29 21:45:34 +00003498 g:StackTop()
Bram Moolenaardfa3d552020-09-10 22:05:08 +02003499 catch
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02003500 v:throwpoint->assert_match('Test_callstack_def\[2\]..StackTop\[4\]..StackBot, line 2')
Bram Moolenaardfa3d552020-09-10 22:05:08 +02003501 endtry
3502enddef
3503
Bram Moolenaare8211a32020-10-09 22:04:29 +02003504" Re-using spot for variable used in block
3505def Test_block_scoped_var()
3506 var lines =<< trim END
3507 vim9script
3508 def Func()
3509 var x = ['a', 'b', 'c']
3510 if 1
3511 var y = 'x'
Bram Moolenaarbb8a7ce2021-04-10 20:10:26 +02003512 map(x, (_, _) => y)
Bram Moolenaare8211a32020-10-09 22:04:29 +02003513 endif
3514 var z = x
3515 assert_equal(['x', 'x', 'x'], z)
3516 enddef
3517 Func()
3518 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003519 v9.CheckScriptSuccess(lines)
Bram Moolenaare8211a32020-10-09 22:04:29 +02003520enddef
3521
Bram Moolenaareeece9e2020-11-20 19:26:48 +01003522def Test_reset_did_emsg()
3523 var lines =<< trim END
3524 @s = 'blah'
3525 au BufWinLeave * #
3526 def Func()
3527 var winid = popup_create('popup', {})
3528 exe '*s'
3529 popup_close(winid)
3530 enddef
3531 Func()
3532 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003533 v9.CheckScriptFailure(lines, 'E492:', 8)
Bram Moolenaar2d870f82020-12-05 13:41:01 +01003534 delfunc! g:Func
Bram Moolenaareeece9e2020-11-20 19:26:48 +01003535enddef
3536
Bram Moolenaar57f799e2020-12-12 20:42:19 +01003537def Test_did_emsg_reset()
3538 # executing an autocommand resets did_emsg, this should not result in a
3539 # builtin function considered failing
3540 var lines =<< trim END
3541 vim9script
3542 au BufWinLeave * #
3543 def Func()
Bram Moolenaar767034c2021-04-09 17:24:52 +02003544 popup_menu('', {callback: (a, b) => popup_create('', {})->popup_close()})
Bram Moolenaar57f799e2020-12-12 20:42:19 +01003545 eval [][0]
3546 enddef
3547 nno <F3> <cmd>call <sid>Func()<cr>
3548 feedkeys("\<F3>\e", 'xt')
3549 END
3550 writefile(lines, 'XemsgReset')
3551 assert_fails('so XemsgReset', ['E684:', 'E684:'], lines, 2)
3552 delete('XemsgReset')
3553 nunmap <F3>
3554 au! BufWinLeave
3555enddef
3556
Bram Moolenaar56602ba2020-12-05 21:22:08 +01003557def Test_abort_with_silent_call()
3558 var lines =<< trim END
3559 vim9script
3560 g:result = 'none'
3561 def Func()
3562 g:result += 3
3563 g:result = 'yes'
3564 enddef
3565 # error is silenced, but function aborts on error
3566 silent! Func()
3567 assert_equal('none', g:result)
3568 unlet g:result
3569 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003570 v9.CheckScriptSuccess(lines)
Bram Moolenaar56602ba2020-12-05 21:22:08 +01003571enddef
3572
Bram Moolenaarf665e972020-12-05 19:17:16 +01003573def Test_continues_with_silent_error()
3574 var lines =<< trim END
3575 vim9script
3576 g:result = 'none'
3577 def Func()
3578 silent! g:result += 3
3579 g:result = 'yes'
3580 enddef
3581 # error is silenced, function does not abort
3582 Func()
3583 assert_equal('yes', g:result)
3584 unlet g:result
3585 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003586 v9.CheckScriptSuccess(lines)
Bram Moolenaarf665e972020-12-05 19:17:16 +01003587enddef
3588
Bram Moolenaaraf0df472020-12-02 20:51:22 +01003589def Test_abort_even_with_silent()
3590 var lines =<< trim END
3591 vim9script
3592 g:result = 'none'
3593 def Func()
3594 eval {-> ''}() .. '' .. {}['X']
3595 g:result = 'yes'
3596 enddef
Bram Moolenaarf665e972020-12-05 19:17:16 +01003597 silent! Func()
Bram Moolenaaraf0df472020-12-02 20:51:22 +01003598 assert_equal('none', g:result)
Bram Moolenaar4029cab2020-12-05 18:13:27 +01003599 unlet g:result
3600 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003601 v9.CheckScriptSuccess(lines)
Bram Moolenaar4029cab2020-12-05 18:13:27 +01003602enddef
3603
Bram Moolenaarf665e972020-12-05 19:17:16 +01003604def Test_cmdmod_silent_restored()
3605 var lines =<< trim END
3606 vim9script
3607 def Func()
3608 g:result = 'none'
3609 silent! g:result += 3
3610 g:result = 'none'
3611 g:result += 3
3612 enddef
3613 Func()
3614 END
3615 # can't use CheckScriptFailure, it ignores the :silent!
3616 var fname = 'Xdefsilent'
3617 writefile(lines, fname)
3618 var caught = 'no'
3619 try
3620 exe 'source ' .. fname
3621 catch /E1030:/
3622 caught = 'yes'
3623 assert_match('Func, line 4', v:throwpoint)
3624 endtry
3625 assert_equal('yes', caught)
3626 delete(fname)
3627enddef
3628
Bram Moolenaar2fecb532021-03-24 22:00:56 +01003629def Test_cmdmod_silent_nested()
3630 var lines =<< trim END
3631 vim9script
3632 var result = ''
3633
3634 def Error()
3635 result ..= 'Eb'
3636 eval [][0]
3637 result ..= 'Ea'
3638 enddef
3639
3640 def Crash()
3641 result ..= 'Cb'
3642 sil! Error()
3643 result ..= 'Ca'
3644 enddef
3645
3646 Crash()
3647 assert_equal('CbEbEaCa', result)
3648 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003649 v9.CheckScriptSuccess(lines)
Bram Moolenaar2fecb532021-03-24 22:00:56 +01003650enddef
3651
Bram Moolenaar4029cab2020-12-05 18:13:27 +01003652def Test_dict_member_with_silent()
3653 var lines =<< trim END
3654 vim9script
3655 g:result = 'none'
3656 var d: dict<any>
3657 def Func()
3658 try
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01003659 g:result = map([], (_, v) => ({}[v]))->join() .. d['']
Bram Moolenaar4029cab2020-12-05 18:13:27 +01003660 catch
3661 endtry
3662 enddef
3663 silent! Func()
3664 assert_equal('0', g:result)
3665 unlet g:result
Bram Moolenaaraf0df472020-12-02 20:51:22 +01003666 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003667 v9.CheckScriptSuccess(lines)
Bram Moolenaaraf0df472020-12-02 20:51:22 +01003668enddef
3669
Bram Moolenaarf9041332021-01-21 19:41:16 +01003670def Test_skip_cmds_with_silent()
3671 var lines =<< trim END
3672 vim9script
3673
3674 def Func(b: bool)
3675 Crash()
3676 enddef
3677
3678 def Crash()
3679 sil! :/not found/d _
3680 sil! :/not found/put _
3681 enddef
3682
3683 Func(true)
3684 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003685 v9.CheckScriptSuccess(lines)
Bram Moolenaarf9041332021-01-21 19:41:16 +01003686enddef
3687
Bram Moolenaar5b3d1bb2020-12-22 12:20:08 +01003688def Test_opfunc()
Bram Moolenaar848fadd2022-01-30 15:28:30 +00003689 nnoremap <F3> <cmd>set opfunc=g:Opfunc<cr>g@
Bram Moolenaar5b3d1bb2020-12-22 12:20:08 +01003690 def g:Opfunc(_: any): string
3691 setline(1, 'ASDF')
3692 return ''
3693 enddef
3694 new
3695 setline(1, 'asdf')
3696 feedkeys("\<F3>$", 'x')
3697 assert_equal('ASDF', getline(1))
3698
3699 bwipe!
3700 nunmap <F3>
3701enddef
3702
Bram Moolenaar3b309f12021-12-13 18:19:55 +00003703func Test_opfunc_error()
3704 CheckScreendump
3705 call Run_Test_opfunc_error()
3706endfunc
3707
3708def Run_Test_opfunc_error()
3709 # test that the error from Opfunc() is displayed right away
3710 var lines =<< trim END
3711 vim9script
3712
3713 def Opfunc(type: string)
3714 try
3715 eval [][0]
3716 catch /nothing/ # error not caught
3717 endtry
3718 enddef
3719 &operatorfunc = Opfunc
3720 nnoremap <expr> l <SID>L()
3721 def L(): string
3722 return 'l'
3723 enddef
3724 'x'->repeat(10)->setline(1)
3725 feedkeys('g@l', 'n')
3726 feedkeys('llll')
3727 END
3728 call writefile(lines, 'XTest_opfunc_error')
3729
Bram Moolenaar62aec932022-01-29 21:45:34 +00003730 var buf = g:RunVimInTerminal('-S XTest_opfunc_error', {rows: 6, wait_for_ruler: 0})
3731 g:WaitForAssert(() => assert_match('Press ENTER', term_getline(buf, 6)))
Bram Moolenaarec892232022-05-06 17:53:06 +01003732 g:WaitForAssert(() => assert_match('E684: List index out of range: 0', term_getline(buf, 5)))
Bram Moolenaar3b309f12021-12-13 18:19:55 +00003733
3734 # clean up
Bram Moolenaar62aec932022-01-29 21:45:34 +00003735 g:StopVimInTerminal(buf)
Bram Moolenaar3b309f12021-12-13 18:19:55 +00003736 delete('XTest_opfunc_error')
3737enddef
3738
Bram Moolenaar077a4232020-12-22 18:33:27 +01003739" this was crashing on exit
3740def Test_nested_lambda_in_closure()
3741 var lines =<< trim END
3742 vim9script
Bram Moolenaar227c58a2021-04-28 20:40:44 +02003743 command WriteDone writefile(['Done'], 'XnestedDone')
Bram Moolenaar077a4232020-12-22 18:33:27 +01003744 def Outer()
3745 def g:Inner()
3746 echo map([1, 2, 3], {_, v -> v + 1})
3747 enddef
3748 g:Inner()
3749 enddef
3750 defcompile
Bram Moolenaar227c58a2021-04-28 20:40:44 +02003751 # not reached
Bram Moolenaar077a4232020-12-22 18:33:27 +01003752 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003753 if !g:RunVim([], lines, '--clean -c WriteDone -c quit')
Bram Moolenaar077a4232020-12-22 18:33:27 +01003754 return
3755 endif
3756 assert_equal(['Done'], readfile('XnestedDone'))
3757 delete('XnestedDone')
3758enddef
3759
Bram Moolenaar92368aa2022-02-07 17:50:39 +00003760def Test_nested_closure_funcref()
3761 var lines =<< trim END
3762 vim9script
3763 def Func()
3764 var n: number
3765 def Nested()
3766 ++n
3767 enddef
3768 Nested()
3769 g:result_one = n
3770 var Ref = function(Nested)
3771 Ref()
3772 g:result_two = n
3773 enddef
3774 Func()
3775 END
3776 v9.CheckScriptSuccess(lines)
3777 assert_equal(1, g:result_one)
3778 assert_equal(2, g:result_two)
3779 unlet g:result_one g:result_two
3780enddef
3781
Bram Moolenaar7aca5ca2022-02-07 19:56:43 +00003782def Test_nested_closure_in_dict()
3783 var lines =<< trim END
3784 vim9script
3785 def Func(): dict<any>
3786 var n: number
3787 def Inc(): number
3788 ++n
3789 return n
3790 enddef
3791 return {inc: function(Inc)}
3792 enddef
3793 disas Func
3794 var d = Func()
3795 assert_equal(1, d.inc())
3796 assert_equal(2, d.inc())
3797 END
3798 v9.CheckScriptSuccess(lines)
3799enddef
3800
Bram Moolenaarfb43cfc2022-03-11 18:54:17 +00003801def Test_script_local_other_script()
3802 var lines =<< trim END
3803 function LegacyJob()
3804 let FuncRef = function('s:close_cb')
3805 endfunction
3806 function s:close_cb(...)
3807 endfunction
3808 END
3809 lines->writefile('Xlegacy.vim')
3810 source Xlegacy.vim
3811 g:LegacyJob()
3812 g:LegacyJob()
3813 g:LegacyJob()
3814
3815 delfunc g:LegacyJob
3816 delete('Xlegacy.vim')
3817enddef
3818
Bram Moolenaar04947cc2021-03-06 19:26:46 +01003819def Test_check_func_arg_types()
3820 var lines =<< trim END
3821 vim9script
3822 def F1(x: string): string
3823 return x
3824 enddef
3825
3826 def F2(x: number): number
3827 return x + 1
3828 enddef
3829
Bram Moolenaar1d9cef72022-03-17 16:30:03 +00003830 def G(Fg: func): dict<func>
3831 return {f: Fg}
Bram Moolenaar04947cc2021-03-06 19:26:46 +01003832 enddef
3833
3834 def H(d: dict<func>): string
3835 return d.f('a')
3836 enddef
3837 END
3838
Bram Moolenaar62aec932022-01-29 21:45:34 +00003839 v9.CheckScriptSuccess(lines + ['echo H(G(F1))'])
3840 v9.CheckScriptFailure(lines + ['echo H(G(F2))'], 'E1013:')
Bram Moolenaar1d9cef72022-03-17 16:30:03 +00003841
3842 v9.CheckScriptFailure(lines + ['def SomeFunc(ff: func)', 'enddef'], 'E704:')
Bram Moolenaar04947cc2021-03-06 19:26:46 +01003843enddef
3844
Bram Moolenaarbadf04f2022-03-12 21:28:22 +00003845def Test_call_func_with_null()
3846 var lines =<< trim END
3847 def Fstring(v: string)
3848 assert_equal(null_string, v)
3849 enddef
3850 Fstring(null_string)
3851 def Fblob(v: blob)
3852 assert_equal(null_blob, v)
3853 enddef
3854 Fblob(null_blob)
3855 def Flist(v: list<number>)
3856 assert_equal(null_list, v)
3857 enddef
3858 Flist(null_list)
3859 def Fdict(v: dict<number>)
3860 assert_equal(null_dict, v)
3861 enddef
3862 Fdict(null_dict)
3863 def Ffunc(Fv: func(number): number)
3864 assert_equal(null_function, Fv)
3865 enddef
3866 Ffunc(null_function)
3867 if has('channel')
3868 def Fchannel(v: channel)
3869 assert_equal(null_channel, v)
3870 enddef
3871 Fchannel(null_channel)
3872 def Fjob(v: job)
3873 assert_equal(null_job, v)
3874 enddef
3875 Fjob(null_job)
3876 endif
3877 END
3878 v9.CheckDefAndScriptSuccess(lines)
3879enddef
3880
3881def Test_null_default_argument()
3882 var lines =<< trim END
3883 def Fstring(v: string = null_string)
3884 assert_equal(null_string, v)
3885 enddef
3886 Fstring()
3887 def Fblob(v: blob = null_blob)
3888 assert_equal(null_blob, v)
3889 enddef
3890 Fblob()
3891 def Flist(v: list<number> = null_list)
3892 assert_equal(null_list, v)
3893 enddef
3894 Flist()
3895 def Fdict(v: dict<number> = null_dict)
3896 assert_equal(null_dict, v)
3897 enddef
3898 Fdict()
3899 def Ffunc(Fv: func(number): number = null_function)
3900 assert_equal(null_function, Fv)
3901 enddef
3902 Ffunc()
3903 if has('channel')
3904 def Fchannel(v: channel = null_channel)
3905 assert_equal(null_channel, v)
3906 enddef
3907 Fchannel()
3908 def Fjob(v: job = null_job)
3909 assert_equal(null_job, v)
3910 enddef
3911 Fjob()
3912 endif
3913 END
3914 v9.CheckDefAndScriptSuccess(lines)
3915enddef
3916
3917def Test_null_return()
3918 var lines =<< trim END
3919 def Fstring(): string
3920 return null_string
3921 enddef
3922 assert_equal(null_string, Fstring())
3923 def Fblob(): blob
3924 return null_blob
3925 enddef
3926 assert_equal(null_blob, Fblob())
3927 def Flist(): list<number>
3928 return null_list
3929 enddef
3930 assert_equal(null_list, Flist())
3931 def Fdict(): dict<number>
3932 return null_dict
3933 enddef
3934 assert_equal(null_dict, Fdict())
3935 def Ffunc(): func(number): number
3936 return null_function
3937 enddef
3938 assert_equal(null_function, Ffunc())
3939 if has('channel')
3940 def Fchannel(): channel
3941 return null_channel
3942 enddef
3943 assert_equal(null_channel, Fchannel())
3944 def Fjob(): job
3945 return null_job
3946 enddef
3947 assert_equal(null_job, Fjob())
3948 endif
3949 END
3950 v9.CheckDefAndScriptSuccess(lines)
3951enddef
3952
Bram Moolenaar6e48b842021-08-10 22:52:02 +02003953def Test_list_any_type_checked()
3954 var lines =<< trim END
3955 vim9script
3956 def Foo()
3957 --decl--
3958 Bar(l)
3959 enddef
3960 def Bar(ll: list<dict<any>>)
3961 enddef
3962 Foo()
3963 END
Bram Moolenaar2d3ac2e2022-02-03 12:34:05 +00003964 # "any" could be "dict<any>", thus OK
Bram Moolenaar6e48b842021-08-10 22:52:02 +02003965 lines[2] = 'var l: list<any>'
Bram Moolenaar2d3ac2e2022-02-03 12:34:05 +00003966 v9.CheckScriptSuccess(lines)
Bram Moolenaar6e48b842021-08-10 22:52:02 +02003967 lines[2] = 'var l: list<any> = []'
Bram Moolenaar2d3ac2e2022-02-03 12:34:05 +00003968 v9.CheckScriptSuccess(lines)
Bram Moolenaar6e48b842021-08-10 22:52:02 +02003969
3970 lines[2] = 'var l: list<any> = [11]'
Bram Moolenaar62aec932022-01-29 21:45:34 +00003971 v9.CheckScriptFailure(lines, 'E1013: Argument 1: type mismatch, expected list<dict<any>> but got list<number>', 2)
Bram Moolenaar6e48b842021-08-10 22:52:02 +02003972enddef
3973
Bram Moolenaar701cc6c2021-04-10 13:33:48 +02003974def Test_compile_error()
3975 var lines =<< trim END
3976 def g:Broken()
3977 echo 'a' + {}
3978 enddef
3979 call g:Broken()
3980 END
3981 # First call: compilation error
Bram Moolenaar62aec932022-01-29 21:45:34 +00003982 v9.CheckScriptFailure(lines, 'E1051: Wrong argument type for +')
Bram Moolenaar701cc6c2021-04-10 13:33:48 +02003983
3984 # Second call won't try compiling again
3985 assert_fails('call g:Broken()', 'E1091: Function is not compiled: Broken')
Bram Moolenaar599410c2021-04-10 14:03:43 +02003986 delfunc g:Broken
3987
3988 # No error when compiling with :silent!
3989 lines =<< trim END
3990 def g:Broken()
3991 echo 'a' + []
3992 enddef
3993 silent! defcompile
3994 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003995 v9.CheckScriptSuccess(lines)
Bram Moolenaar599410c2021-04-10 14:03:43 +02003996
3997 # Calling the function won't try compiling again
3998 assert_fails('call g:Broken()', 'E1091: Function is not compiled: Broken')
3999 delfunc g:Broken
Bram Moolenaar701cc6c2021-04-10 13:33:48 +02004000enddef
4001
Bram Moolenaar962c43b2021-04-10 17:18:09 +02004002def Test_ignored_argument()
4003 var lines =<< trim END
4004 vim9script
4005 def Ignore(_, _): string
4006 return 'yes'
4007 enddef
4008 assert_equal('yes', Ignore(1, 2))
4009
4010 func Ok(_)
4011 return a:_
4012 endfunc
4013 assert_equal('ok', Ok('ok'))
4014
4015 func Oktoo()
4016 let _ = 'too'
4017 return _
4018 endfunc
4019 assert_equal('too', Oktoo())
Bram Moolenaarda479c72021-04-10 21:01:38 +02004020
4021 assert_equal([[1], [2], [3]], range(3)->mapnew((_, v) => [v]->map((_, w) => w + 1)))
Bram Moolenaar962c43b2021-04-10 17:18:09 +02004022 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00004023 v9.CheckScriptSuccess(lines)
Bram Moolenaar962c43b2021-04-10 17:18:09 +02004024
4025 lines =<< trim END
4026 def Ignore(_: string): string
4027 return _
4028 enddef
4029 defcompile
4030 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00004031 v9.CheckScriptFailure(lines, 'E1181:', 1)
Bram Moolenaar962c43b2021-04-10 17:18:09 +02004032
4033 lines =<< trim END
4034 var _ = 1
4035 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00004036 v9.CheckDefAndScriptFailure(lines, 'E1181:', 1)
Yegappan Lakshmanan34fcb692021-05-25 20:14:00 +02004037
4038 lines =<< trim END
4039 var x = _
4040 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00004041 v9.CheckDefAndScriptFailure(lines, 'E1181:', 1)
Bram Moolenaar962c43b2021-04-10 17:18:09 +02004042enddef
4043
Bram Moolenaarbb8a7ce2021-04-10 20:10:26 +02004044def Test_too_many_arguments()
4045 var lines =<< trim END
4046 echo [0, 1, 2]->map(() => 123)
4047 END
Bram Moolenaareddd4fc2022-02-20 15:52:28 +00004048 v9.CheckDefAndScriptFailure(lines, ['E176:', 'E1106: 2 arguments too many'], 1)
Bram Moolenaarbb8a7ce2021-04-10 20:10:26 +02004049
4050 lines =<< trim END
4051 echo [0, 1, 2]->map((_) => 123)
4052 END
Bram Moolenaareddd4fc2022-02-20 15:52:28 +00004053 v9.CheckDefAndScriptFailure(lines, ['E176', 'E1106: One argument too many'], 1)
Bram Moolenaarbb8a7ce2021-04-10 20:10:26 +02004054enddef
Bram Moolenaar077a4232020-12-22 18:33:27 +01004055
Bram Moolenaara6aa1642021-04-23 19:32:23 +02004056def Test_closing_brace_at_start_of_line()
4057 var lines =<< trim END
4058 def Func()
4059 enddef
4060 Func(
4061 )
4062 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00004063 v9.CheckDefAndScriptSuccess(lines)
Bram Moolenaara6aa1642021-04-23 19:32:23 +02004064enddef
4065
Bram Moolenaar62aec932022-01-29 21:45:34 +00004066func s:CreateMydict()
Bram Moolenaarb033ee22021-08-15 16:08:36 +02004067 let g:mydict = {}
4068 func g:mydict.afunc()
4069 let g:result = self.key
4070 endfunc
4071endfunc
4072
4073def Test_numbered_function_reference()
4074 CreateMydict()
4075 var output = execute('legacy func g:mydict.afunc')
4076 var funcName = 'g:' .. substitute(output, '.*function \(\d\+\).*', '\1', '')
4077 execute 'function(' .. funcName .. ', [], {key: 42})()'
4078 # check that the function still exists
4079 assert_equal(output, execute('legacy func g:mydict.afunc'))
4080 unlet g:mydict
4081enddef
4082
Bram Moolenaard3a11782022-01-05 16:50:40 +00004083def Test_go_beyond_end_of_cmd()
4084 # this was reading the byte after the end of the line
4085 var lines =<< trim END
4086 def F()
4087 cal
4088 enddef
4089 defcompile
4090 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00004091 v9.CheckScriptFailure(lines, 'E476:')
Bram Moolenaard3a11782022-01-05 16:50:40 +00004092enddef
4093
Yegappan Lakshmanan7c7e19c2022-04-09 11:09:07 +01004094" Test for memory allocation failure when defining a new lambda
4095func Test_lambda_allocation_failure()
4096 new
4097 let lines =<< trim END
4098 vim9script
4099 g:Xlambda = (x): number => {
4100 return x + 1
4101 }
4102 END
4103 call setline(1, lines)
4104 call test_alloc_fail(GetAllocId('get_func'), 0, 0)
4105 call assert_fails('source', 'E342:')
4106 call assert_false(exists('g:Xlambda'))
4107 bw!
4108endfunc
4109
Bram Moolenaar8b716f52022-02-15 21:17:56 +00004110" The following messes up syntax highlight, keep near the end.
Bram Moolenaar20677332021-06-06 17:02:53 +02004111if has('python3')
Bram Moolenaar8b716f52022-02-15 21:17:56 +00004112 def Test_python3_command()
4113 py3 import vim
Bram Moolenaarf5288c52022-02-15 21:33:29 +00004114 py3 vim.command("g:done = 'yes'")
Bram Moolenaar8b716f52022-02-15 21:17:56 +00004115 assert_equal('yes', g:done)
4116 unlet g:done
4117 enddef
4118
Bram Moolenaar20677332021-06-06 17:02:53 +02004119 def Test_python3_heredoc()
4120 py3 << trim EOF
4121 import vim
4122 vim.vars['didit'] = 'yes'
4123 EOF
4124 assert_equal('yes', g:didit)
4125
4126 python3 << trim EOF
4127 import vim
4128 vim.vars['didit'] = 'again'
4129 EOF
4130 assert_equal('again', g:didit)
4131 enddef
4132endif
4133
Bram Moolenaar20677332021-06-06 17:02:53 +02004134if has('lua')
4135 def Test_lua_heredoc()
4136 g:d = {}
4137 lua << trim EOF
4138 x = vim.eval('g:d')
4139 x['key'] = 'val'
4140 EOF
4141 assert_equal('val', g:d.key)
4142 enddef
Bram Moolenaarefd73ae2022-03-20 18:51:00 +00004143
4144 def Test_lua_heredoc_fails()
4145 var lines = [
4146 'vim9script',
4147 'def ExeLua()',
4148 'lua << trim EOLUA',
4149 "x = vim.eval('g:nodict')",
4150 'EOLUA',
4151 'enddef',
4152 'ExeLua()',
4153 ]
4154 v9.CheckScriptFailure(lines, 'E121: Undefined variable: g:nodict')
4155 enddef
Bram Moolenaar20677332021-06-06 17:02:53 +02004156endif
4157
Bram Moolenaarf7779c62020-05-03 15:38:16 +02004158
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02004159" vim: ts=8 sw=2 sts=2 expandtab tw=80 fdm=marker