blob: a5e3e902548f56142b19e5721209918155ff3424 [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 Moolenaarf5fec052022-09-11 11:49:22 +010032 writefile(lines, 'XTest_compile_error', 'D')
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 +020040enddef
41
42def TestCompilingErrorInTry()
Bram Moolenaar3b0d70f2022-08-29 22:31:20 +010043 var dir = 'Xcompdir/autoload'
Bram Moolenaarf5fec052022-09-11 11:49:22 +010044 mkdir(dir, 'pR')
Bram Moolenaare8c46602021-04-05 22:27:37 +020045
46 var lines =<< trim END
47 vim9script
Bram Moolenaard8fe6d32022-01-30 18:40:44 +000048 export def OnlyCompiled()
Bram Moolenaare8c46602021-04-05 22:27:37 +020049 g:runtime = 'yes'
50 invalid
51 enddef
52 END
53 writefile(lines, dir .. '/script.vim')
54
55 lines =<< trim END
56 vim9script
57 todo
58 try
59 script#OnlyCompiled()
60 catch /nothing/
61 endtry
62 END
Bram Moolenaar3b0d70f2022-08-29 22:31:20 +010063 lines[1] = 'set rtp=' .. getcwd() .. '/Xcompdir'
Bram Moolenaarf5fec052022-09-11 11:49:22 +010064 writefile(lines, 'XTest_compile_error', 'D')
Bram Moolenaare8c46602021-04-05 22:27:37 +020065
Bram Moolenaar62aec932022-01-29 21:45:34 +000066 var buf = g:RunVimInTerminal('-S XTest_compile_error', {rows: 10, wait_for_ruler: 0})
67 g:WaitForAssert(() => assert_match('Error detected while compiling command line.*function script#OnlyCompiled.*Invalid command: invalid',
68 g:Term_getlines(buf, range(1, 9))))
Bram Moolenaare8c46602021-04-05 22:27:37 +020069
70 # clean up
Bram Moolenaar62aec932022-01-29 21:45:34 +000071 g:StopVimInTerminal(buf)
Bram Moolenaarf4e8cdd2020-10-12 22:07:13 +020072enddef
73
Bram Moolenaarad6d9cc2022-08-08 21:43:11 +010074def Test_comment_error()
75 v9.CheckDefFailure(['#{ comment'], 'E1170:')
76enddef
77
Bram Moolenaarb55d6182021-06-08 22:01:53 +020078def Test_compile_error_in_called_function()
79 var lines =<< trim END
80 vim9script
81 var n: number
82 def Foo()
83 &hls = n
84 enddef
85 def Bar()
86 Foo()
87 enddef
88 silent! Foo()
89 Bar()
90 END
Bram Moolenaar62aec932022-01-29 21:45:34 +000091 v9.CheckScriptFailureList(lines, ['E1012:', 'E1191:'])
Bram Moolenaarb55d6182021-06-08 22:01:53 +020092enddef
93
Bram Moolenaar22f17a22021-06-21 20:48:58 +020094def Test_wrong_function_name()
95 var lines =<< trim END
96 vim9script
97 func _Foo()
98 echo 'foo'
99 endfunc
100 END
Bram Moolenaar3787f262022-02-07 21:54:01 +0000101 v9.CheckScriptFailure(lines, 'E1267:')
Bram Moolenaar22f17a22021-06-21 20:48:58 +0200102
103 lines =<< trim END
104 vim9script
105 def _Foo()
106 echo 'foo'
107 enddef
108 END
Bram Moolenaar3787f262022-02-07 21:54:01 +0000109 v9.CheckScriptFailure(lines, 'E1267:')
Bram Moolenaardea5ab02022-02-23 22:12:02 +0000110
111 lines =<< trim END
112 vim9script
113 var Object = {}
114 function Object.Method()
115 endfunction
116 END
117 v9.CheckScriptFailure(lines, 'E1182:')
118
119 lines =<< trim END
120 vim9script
121 var Object = {}
122 def Object.Method()
123 enddef
124 END
125 v9.CheckScriptFailure(lines, 'E1182:')
126
127 lines =<< trim END
128 vim9script
129 g:Object = {}
130 function g:Object.Method()
131 endfunction
132 END
133 v9.CheckScriptFailure(lines, 'E1182:')
134
135 lines =<< trim END
136 let s:Object = {}
137 def Define()
138 function s:Object.Method()
139 endfunction
140 enddef
141 defcompile
142 END
143 v9.CheckScriptFailure(lines, 'E1182:')
144 delfunc g:Define
145
146 lines =<< trim END
147 let s:Object = {}
148 def Define()
149 def Object.Method()
150 enddef
151 enddef
152 defcompile
153 END
154 v9.CheckScriptFailure(lines, 'E1182:')
155 delfunc g:Define
156
157 lines =<< trim END
158 let g:Object = {}
159 def Define()
160 function g:Object.Method()
161 endfunction
162 enddef
163 defcompile
164 END
165 v9.CheckScriptFailure(lines, 'E1182:')
166 delfunc g:Define
Bram Moolenaar22f17a22021-06-21 20:48:58 +0200167enddef
168
Bram Moolenaarf48b2fa2021-04-12 22:02:36 +0200169def Test_autoload_name_mismatch()
Bram Moolenaar3b0d70f2022-08-29 22:31:20 +0100170 var dir = 'Xnamedir/autoload'
Bram Moolenaarf5fec052022-09-11 11:49:22 +0100171 mkdir(dir, 'pR')
Bram Moolenaarf48b2fa2021-04-12 22:02:36 +0200172
173 var lines =<< trim END
174 vim9script
Bram Moolenaard8fe6d32022-01-30 18:40:44 +0000175 export def NoFunction()
Bram Moolenaarf48b2fa2021-04-12 22:02:36 +0200176 # comment
177 g:runtime = 'yes'
178 enddef
179 END
180 writefile(lines, dir .. '/script.vim')
181
182 var save_rtp = &rtp
Bram Moolenaar3b0d70f2022-08-29 22:31:20 +0100183 exe 'set rtp=' .. getcwd() .. '/Xnamedir'
Bram Moolenaarf48b2fa2021-04-12 22:02:36 +0200184 lines =<< trim END
185 call script#Function()
186 END
Bram Moolenaard8fe6d32022-01-30 18:40:44 +0000187 v9.CheckScriptFailure(lines, 'E117:', 1)
Bram Moolenaarf48b2fa2021-04-12 22:02:36 +0200188
189 &rtp = save_rtp
Bram Moolenaarf48b2fa2021-04-12 22:02:36 +0200190enddef
191
Bram Moolenaarf0a40692021-06-11 22:05:47 +0200192def Test_autoload_names()
Bram Moolenaar3b0d70f2022-08-29 22:31:20 +0100193 var dir = 'Xandir/autoload'
Bram Moolenaarf5fec052022-09-11 11:49:22 +0100194 mkdir(dir, 'pR')
Bram Moolenaarf0a40692021-06-11 22:05:47 +0200195
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
Bram Moolenaar3b0d70f2022-08-29 22:31:20 +0100205 exe 'set rtp=' .. getcwd() .. '/Xandir'
Bram Moolenaarf0a40692021-06-11 22:05:47 +0200206
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
Bram Moolenaarf0a40692021-06-11 22:05:47 +0200217enddef
218
Bram Moolenaar88c89c72021-08-14 14:01:05 +0200219def Test_autoload_error_in_script()
Bram Moolenaar3b0d70f2022-08-29 22:31:20 +0100220 var dir = 'Xaedir/autoload'
Bram Moolenaarf5fec052022-09-11 11:49:22 +0100221 mkdir(dir, 'pR')
Bram Moolenaar88c89c72021-08-14 14:01:05 +0200222
223 var lines =<< trim END
224 func scripterror#function()
225 let g:called_function = 'yes'
226 endfunc
227 let 0 = 1
228 END
229 writefile(lines, dir .. '/scripterror.vim')
230
231 var save_rtp = &rtp
Bram Moolenaar3b0d70f2022-08-29 22:31:20 +0100232 exe 'set rtp=' .. getcwd() .. '/Xaedir'
Bram Moolenaar88c89c72021-08-14 14:01:05 +0200233
234 g:called_function = 'no'
235 # The error in the autoload script cannot be checked with assert_fails(), use
236 # CheckDefSuccess() instead of CheckDefFailure()
237 try
Bram Moolenaar62aec932022-01-29 21:45:34 +0000238 v9.CheckDefSuccess(['scripterror#function()'])
Bram Moolenaar88c89c72021-08-14 14:01:05 +0200239 catch
240 assert_match('E121: Undefined variable: 0', v:exception)
241 endtry
242 assert_equal('no', g:called_function)
243
244 lines =<< trim END
245 func scriptcaught#function()
246 let g:called_function = 'yes'
247 endfunc
248 try
249 let 0 = 1
250 catch
251 let g:caught = v:exception
252 endtry
253 END
254 writefile(lines, dir .. '/scriptcaught.vim')
255
256 g:called_function = 'no'
Bram Moolenaar62aec932022-01-29 21:45:34 +0000257 v9.CheckDefSuccess(['scriptcaught#function()'])
Bram Moolenaar88c89c72021-08-14 14:01:05 +0200258 assert_match('E121: Undefined variable: 0', g:caught)
259 assert_equal('yes', g:called_function)
260
261 &rtp = save_rtp
Bram Moolenaar88c89c72021-08-14 14:01:05 +0200262enddef
263
Bram Moolenaar62aec932022-01-29 21:45:34 +0000264def s:CallRecursive(n: number): number
Bram Moolenaar0ba48e82020-11-17 18:23:19 +0100265 return CallRecursive(n + 1)
266enddef
267
Bram Moolenaar62aec932022-01-29 21:45:34 +0000268def s:CallMapRecursive(l: list<number>): number
Bram Moolenaar2949cfd2020-12-31 21:28:47 +0100269 return map(l, (_, v) => CallMapRecursive([v]))[0]
Bram Moolenaar0ba48e82020-11-17 18:23:19 +0100270enddef
271
272def Test_funcdepth_error()
273 set maxfuncdepth=10
274
275 var caught = false
276 try
277 CallRecursive(1)
278 catch /E132:/
279 caught = true
280 endtry
281 assert_true(caught)
282
283 caught = false
284 try
285 CallMapRecursive([1])
286 catch /E132:/
287 caught = true
288 endtry
289 assert_true(caught)
290
291 set maxfuncdepth&
292enddef
293
Bram Moolenaar5178b1b2021-01-01 18:43:51 +0100294def Test_endfunc_enddef()
295 var lines =<< trim END
296 def Test()
297 echo 'test'
298 endfunc
299 enddef
300 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000301 v9.CheckScriptFailure(lines, 'E1151:', 3)
Bram Moolenaar5178b1b2021-01-01 18:43:51 +0100302
303 lines =<< trim END
304 def Test()
305 func Nested()
306 echo 'test'
307 enddef
308 enddef
309 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000310 v9.CheckScriptFailure(lines, 'E1152:', 4)
Bram Moolenaar49f1e9e2021-03-22 20:49:02 +0100311
312 lines =<< trim END
313 def Ok()
314 echo 'hello'
315 enddef | echo 'there'
316 def Bad()
317 echo 'hello'
318 enddef there
319 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000320 v9.CheckScriptFailure(lines, 'E1173: Text found after enddef: there', 6)
Bram Moolenaar5178b1b2021-01-01 18:43:51 +0100321enddef
322
Bram Moolenaarb8ba9b92021-01-01 18:54:34 +0100323def Test_missing_endfunc_enddef()
324 var lines =<< trim END
325 vim9script
326 def Test()
327 echo 'test'
328 endef
329 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000330 v9.CheckScriptFailure(lines, 'E1057:', 2)
Bram Moolenaarb8ba9b92021-01-01 18:54:34 +0100331
332 lines =<< trim END
333 vim9script
334 func Some()
335 echo 'test'
336 enfffunc
337 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000338 v9.CheckScriptFailure(lines, 'E126:', 2)
Bram Moolenaarb8ba9b92021-01-01 18:54:34 +0100339enddef
340
Bram Moolenaar4efd9942021-01-24 21:14:20 +0100341def Test_white_space_before_paren()
342 var lines =<< trim END
343 vim9script
344 def Test ()
345 echo 'test'
346 enddef
347 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000348 v9.CheckScriptFailure(lines, 'E1068:', 2)
Bram Moolenaar4efd9942021-01-24 21:14:20 +0100349
350 lines =<< trim END
351 vim9script
352 func Test ()
353 echo 'test'
354 endfunc
355 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000356 v9.CheckScriptFailure(lines, 'E1068:', 2)
Bram Moolenaar4efd9942021-01-24 21:14:20 +0100357
358 lines =<< trim END
359 def Test ()
360 echo 'test'
361 enddef
362 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000363 v9.CheckScriptFailure(lines, 'E1068:', 1)
Bram Moolenaar4efd9942021-01-24 21:14:20 +0100364
365 lines =<< trim END
366 func Test ()
367 echo 'test'
368 endfunc
369 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000370 v9.CheckScriptSuccess(lines)
Bram Moolenaar4efd9942021-01-24 21:14:20 +0100371enddef
372
Bram Moolenaar832ea892021-01-08 21:55:26 +0100373def Test_enddef_dict_key()
374 var d = {
375 enddef: 'x',
376 endfunc: 'y',
377 }
378 assert_equal({enddef: 'x', endfunc: 'y'}, d)
379enddef
380
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200381def ReturnString(): string
382 return 'string'
383enddef
384
385def ReturnNumber(): number
386 return 123
387enddef
388
389let g:notNumber = 'string'
390
391def ReturnGlobal(): number
392 return g:notNumber
393enddef
394
395def Test_return_something()
Bram Moolenaar62aec932022-01-29 21:45:34 +0000396 g:ReturnString()->assert_equal('string')
397 g:ReturnNumber()->assert_equal(123)
Bram Moolenaar848fadd2022-01-30 15:28:30 +0000398 assert_fails('g:ReturnGlobal()', 'E1012: Type mismatch; expected number but got string', '', 1, 'ReturnGlobal')
Bram Moolenaaref7aadb2022-01-18 18:46:07 +0000399
400 var lines =<< trim END
401 vim9script
402
403 def Msg()
404 echomsg 'in Msg()...'
405 enddef
406
407 def Func()
408 return Msg()
409 enddef
410 defcompile
411 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000412 v9.CheckScriptFailure(lines, 'E1096:')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200413enddef
414
Bram Moolenaare32e5162021-01-21 20:21:29 +0100415def Test_check_argument_type()
416 var lines =<< trim END
417 vim9script
418 def Val(a: number, b: number): number
419 return 0
420 enddef
421 def Func()
422 var x: any = true
423 Val(0, x)
424 enddef
425 disass Func
426 Func()
427 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000428 v9.CheckScriptFailure(lines, 'E1013: Argument 2: type mismatch, expected number but got bool', 2)
Bram Moolenaare32e5162021-01-21 20:21:29 +0100429enddef
430
Bram Moolenaarefd88552020-06-18 20:50:10 +0200431def Test_missing_return()
Bram Moolenaar62aec932022-01-29 21:45:34 +0000432 v9.CheckDefFailure(['def Missing(): number',
Bram Moolenaarefd88552020-06-18 20:50:10 +0200433 ' if g:cond',
434 ' echo "no return"',
435 ' else',
436 ' return 0',
Bram Moolenaar2984ed32022-08-20 14:51:17 +0100437 ' endif',
Bram Moolenaarefd88552020-06-18 20:50:10 +0200438 'enddef'], 'E1027:')
Bram Moolenaar62aec932022-01-29 21:45:34 +0000439 v9.CheckDefFailure(['def Missing(): number',
Bram Moolenaarefd88552020-06-18 20:50:10 +0200440 ' if g:cond',
441 ' return 1',
442 ' else',
443 ' echo "no return"',
Bram Moolenaar2984ed32022-08-20 14:51:17 +0100444 ' endif',
Bram Moolenaarefd88552020-06-18 20:50:10 +0200445 'enddef'], 'E1027:')
Bram Moolenaar62aec932022-01-29 21:45:34 +0000446 v9.CheckDefFailure(['def Missing(): number',
Bram Moolenaarefd88552020-06-18 20:50:10 +0200447 ' if g:cond',
448 ' return 1',
449 ' else',
450 ' return 2',
Bram Moolenaar2984ed32022-08-20 14:51:17 +0100451 ' endif',
452 ' return 3',
Bram Moolenaarefd88552020-06-18 20:50:10 +0200453 'enddef'], 'E1095:')
454enddef
455
Bram Moolenaar403dc312020-10-17 19:29:51 +0200456def Test_return_bool()
457 var lines =<< trim END
458 vim9script
459 def MenuFilter(id: number, key: string): bool
460 return popup_filter_menu(id, key)
461 enddef
462 def YesnoFilter(id: number, key: string): bool
463 return popup_filter_yesno(id, key)
464 enddef
465 defcompile
466 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000467 v9.CheckScriptSuccess(lines)
Bram Moolenaar403dc312020-10-17 19:29:51 +0200468enddef
469
mityu500c4442022-12-02 18:12:05 +0000470def Test_return_void_comment_follows()
471 var lines =<< trim END
472 vim9script
473 def ReturnCommentFollows(): void
474 return # Some comment
475 enddef
476 defcompile
477 END
478 v9.CheckScriptSuccess(lines)
479enddef
480
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200481let s:nothing = 0
482def ReturnNothing()
483 s:nothing = 1
484 if true
485 return
486 endif
487 s:nothing = 2
488enddef
489
490def Test_return_nothing()
Bram Moolenaar62aec932022-01-29 21:45:34 +0000491 g:ReturnNothing()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +0200492 s:nothing->assert_equal(1)
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200493enddef
494
Bram Moolenaar648ea762021-01-15 19:04:32 +0100495def Test_return_invalid()
496 var lines =<< trim END
497 vim9script
498 def Func(): invalid
499 return xxx
500 enddef
501 defcompile
502 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000503 v9.CheckScriptFailure(lines, 'E1010:', 2)
Bram Moolenaar31842cd2021-02-12 22:10:21 +0100504
505 lines =<< trim END
506 vim9script
507 def Test(Fun: func(number): number): list<number>
508 return map([1, 2, 3], (_, i) => Fun(i))
509 enddef
510 defcompile
511 def Inc(nr: number): nr
512 return nr + 2
513 enddef
514 echo Test(Inc)
515 END
516 # doing this twice was leaking memory
Bram Moolenaar62aec932022-01-29 21:45:34 +0000517 v9.CheckScriptFailure(lines, 'E1010:')
518 v9.CheckScriptFailure(lines, 'E1010:')
Bram Moolenaar648ea762021-01-15 19:04:32 +0100519enddef
520
Bram Moolenaarefc084e2021-09-09 22:30:52 +0200521def Test_return_list_any()
Bram Moolenaar114dbda2022-01-03 12:28:03 +0000522 # This used to fail but now the actual list type is checked, and since it has
523 # an item of type string it can be used as list<string>.
Bram Moolenaarefc084e2021-09-09 22:30:52 +0200524 var lines =<< trim END
525 vim9script
526 def Func(): list<string>
527 var l: list<any>
528 l->add('string')
529 return l
530 enddef
531 echo Func()
532 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000533 v9.CheckScriptSuccess(lines)
Bram Moolenaar114dbda2022-01-03 12:28:03 +0000534
Bram Moolenaarefc084e2021-09-09 22:30:52 +0200535 lines =<< trim END
536 vim9script
537 def Func(): list<string>
538 var l: list<any>
539 l += ['string']
540 return l
541 enddef
542 echo Func()
543 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000544 v9.CheckScriptSuccess(lines)
Bram Moolenaarefc084e2021-09-09 22:30:52 +0200545enddef
546
Bram Moolenaar1a572e92022-03-15 12:28:10 +0000547def Test_return_any_two_types()
548 var lines =<< trim END
549 vim9script
550
551 def G(Fn: func(string): any)
552 g:result = Fn("hello")
553 enddef
554
555 def F(a: number, b: string): any
556 echo b
557 if a > 0
558 return 1
559 else
560 return []
561 endif
562 enddef
563
564 G(function(F, [1]))
565 END
566 v9.CheckScriptSuccess(lines)
567 assert_equal(1, g:result)
568 unlet g:result
569enddef
570
Bram Moolenaar62aec932022-01-29 21:45:34 +0000571func s:Increment()
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200572 let g:counter += 1
573endfunc
574
575def Test_call_ufunc_count()
576 g:counter = 1
577 Increment()
578 Increment()
579 Increment()
Bram Moolenaarf5be8cd2020-07-17 20:36:00 +0200580 # works with and without :call
Bram Moolenaarc0c71e92020-09-11 19:09:48 +0200581 g:counter->assert_equal(4)
582 eval g:counter->assert_equal(4)
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200583 unlet g:counter
584enddef
585
Bram Moolenaar1983f1a2022-02-28 20:55:02 +0000586def Test_call_ufunc_failure()
587 var lines =<< trim END
588 vim9script
589 def Tryit()
590 g:Global(1, 2, 3)
591 enddef
592
593 func g:Global(a, b, c)
594 echo a:a a:b a:c
595 endfunc
596
597 defcompile
598
599 func! g:Global(a, b)
600 echo a:a a:b
601 endfunc
602 Tryit()
603 END
604 v9.CheckScriptFailure(lines, 'E118: Too many arguments for function: Global')
605 delfunc g:Global
606
607 lines =<< trim END
608 vim9script
609
610 g:Ref = function('len')
611 def Tryit()
612 g:Ref('x')
613 enddef
614
615 defcompile
616
617 g:Ref = function('add')
618 Tryit()
619 END
620 v9.CheckScriptFailure(lines, 'E119: Not enough arguments for function: add')
621 unlet g:Ref
622enddef
623
Bram Moolenaar62aec932022-01-29 21:45:34 +0000624def s:MyVarargs(arg: string, ...rest: list<string>): string
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +0200625 var res = arg
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200626 for s in rest
627 res ..= ',' .. s
628 endfor
629 return res
630enddef
631
632def Test_call_varargs()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +0200633 MyVarargs('one')->assert_equal('one')
634 MyVarargs('one', 'two')->assert_equal('one,two')
635 MyVarargs('one', 'two', 'three')->assert_equal('one,two,three')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200636enddef
637
Bram Moolenaar01dd6c32021-09-05 16:36:23 +0200638def Test_call_white_space()
Bram Moolenaar62aec932022-01-29 21:45:34 +0000639 v9.CheckDefAndScriptFailure(["call Test ('text')"], ['E476:', 'E1068:'])
Bram Moolenaar01dd6c32021-09-05 16:36:23 +0200640enddef
641
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200642def MyDefaultArgs(name = 'string'): string
643 return name
644enddef
645
Bram Moolenaar62aec932022-01-29 21:45:34 +0000646def s:MyDefaultSecond(name: string, second: bool = true): string
Bram Moolenaare30f64b2020-07-15 19:48:20 +0200647 return second ? name : 'none'
648enddef
649
Bram Moolenaar38a3bfa2021-03-29 22:14:55 +0200650
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200651def Test_call_default_args()
Bram Moolenaar62aec932022-01-29 21:45:34 +0000652 g:MyDefaultArgs()->assert_equal('string')
653 g:MyDefaultArgs(v:none)->assert_equal('string')
654 g:MyDefaultArgs('one')->assert_equal('one')
655 assert_fails('g:MyDefaultArgs("one", "two")', 'E118:', '', 4, 'Test_call_default_args')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200656
Bram Moolenaarc0c71e92020-09-11 19:09:48 +0200657 MyDefaultSecond('test')->assert_equal('test')
658 MyDefaultSecond('test', true)->assert_equal('test')
659 MyDefaultSecond('test', false)->assert_equal('none')
Bram Moolenaare30f64b2020-07-15 19:48:20 +0200660
Bram Moolenaar38a3bfa2021-03-29 22:14:55 +0200661 var lines =<< trim END
662 def MyDefaultThird(name: string, aa = 'aa', bb = 'bb'): string
663 return name .. aa .. bb
664 enddef
665
666 MyDefaultThird('->')->assert_equal('->aabb')
667 MyDefaultThird('->', v:none)->assert_equal('->aabb')
668 MyDefaultThird('->', 'xx')->assert_equal('->xxbb')
669 MyDefaultThird('->', v:none, v:none)->assert_equal('->aabb')
670 MyDefaultThird('->', 'xx', v:none)->assert_equal('->xxbb')
671 MyDefaultThird('->', v:none, 'yy')->assert_equal('->aayy')
672 MyDefaultThird('->', 'xx', 'yy')->assert_equal('->xxyy')
Bram Moolenaare28d9b32021-07-03 18:56:53 +0200673
674 def DefArg(mandatory: any, optional = mandatory): string
675 return mandatory .. optional
676 enddef
677 DefArg(1234)->assert_equal('12341234')
678 DefArg("ok")->assert_equal('okok')
Bram Moolenaar38a3bfa2021-03-29 22:14:55 +0200679 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000680 v9.CheckDefAndScriptSuccess(lines)
Bram Moolenaar38a3bfa2021-03-29 22:14:55 +0200681
Bram Moolenaar62aec932022-01-29 21:45:34 +0000682 v9.CheckScriptFailure(['def Func(arg: number = asdf)', 'enddef', 'defcompile'], 'E1001:')
Bram Moolenaar2d870f82020-12-05 13:41:01 +0100683 delfunc g:Func
Bram Moolenaar62aec932022-01-29 21:45:34 +0000684 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 +0100685 delfunc g:Func
Bram Moolenaar62aec932022-01-29 21:45:34 +0000686 v9.CheckDefFailure(['def Func(x: number = )', 'enddef'], 'E15:')
Bram Moolenaar12bce952021-03-11 20:04:04 +0100687
Bram Moolenaar38a3bfa2021-03-29 22:14:55 +0200688 lines =<< trim END
Bram Moolenaar12bce952021-03-11 20:04:04 +0100689 vim9script
690 def Func(a = b == 0 ? 1 : 2, b = 0)
691 enddef
692 defcompile
693 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000694 v9.CheckScriptFailure(lines, 'E1001: Variable not found: b')
Bram Moolenaar59618fe2021-12-21 12:32:17 +0000695
Bram Moolenaarfa46ead2021-12-22 13:18:39 +0000696 # using script variable requires matching type or type cast when executed
Bram Moolenaar59618fe2021-12-21 12:32:17 +0000697 lines =<< trim END
698 vim9script
699 var a: any
700 def Func(arg: string = a)
701 echo arg
702 enddef
703 defcompile
704 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000705 v9.CheckScriptSuccess(lines + ['a = "text"', 'Func()'])
706 v9.CheckScriptFailure(lines + ['a = 123', 'Func()'], 'E1013: Argument 1: type mismatch, expected string but got number')
Bram Moolenaar59618fe2021-12-21 12:32:17 +0000707
708 # using global variable does not require type cast
709 lines =<< trim END
710 vim9script
711 def Func(arg: string = g:str)
712 echo arg
713 enddef
714 g:str = 'works'
715 Func()
716 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000717 v9.CheckScriptSuccess(lines)
Bram Moolenaar04b12692020-05-04 23:24:44 +0200718enddef
719
Bram Moolenaar62aec932022-01-29 21:45:34 +0000720def s:FuncWithComment( # comment
Bram Moolenaarcef12702021-01-04 14:09:43 +0100721 a: number, #comment
722 b: bool, # comment
723 c: string) #comment
724 assert_equal(4, a)
725 assert_equal(true, b)
726 assert_equal('yes', c)
727enddef
728
729def Test_func_with_comments()
730 FuncWithComment(4, true, 'yes')
731
732 var lines =<< trim END
733 def Func(# comment
734 arg: string)
735 enddef
736 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000737 v9.CheckScriptFailure(lines, 'E125:', 1)
Bram Moolenaarcef12702021-01-04 14:09:43 +0100738
739 lines =<< trim END
740 def Func(
741 arg: string# comment
742 )
743 enddef
744 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000745 v9.CheckScriptFailure(lines, 'E475:', 2)
Bram Moolenaarcef12702021-01-04 14:09:43 +0100746
747 lines =<< trim END
748 def Func(
749 arg: string
750 )# comment
751 enddef
752 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000753 v9.CheckScriptFailure(lines, 'E488:', 3)
Bram Moolenaarcef12702021-01-04 14:09:43 +0100754enddef
755
Bram Moolenaar04b12692020-05-04 23:24:44 +0200756def Test_nested_function()
Bram Moolenaar38453522021-11-28 22:00:12 +0000757 def NestedDef(arg: string): string
Bram Moolenaar04b12692020-05-04 23:24:44 +0200758 return 'nested ' .. arg
759 enddef
Bram Moolenaar38453522021-11-28 22:00:12 +0000760 NestedDef(':def')->assert_equal('nested :def')
761
762 func NestedFunc(arg)
763 return 'nested ' .. a:arg
764 endfunc
765 NestedFunc(':func')->assert_equal('nested :func')
Bram Moolenaar04b12692020-05-04 23:24:44 +0200766
Bram Moolenaar62aec932022-01-29 21:45:34 +0000767 v9.CheckDefFailure(['def Nested()', 'enddef', 'Nested(66)'], 'E118:')
768 v9.CheckDefFailure(['def Nested(arg: string)', 'enddef', 'Nested()'], 'E119:')
Bram Moolenaar0e65d3d2020-05-05 17:53:16 +0200769
Bram Moolenaar62aec932022-01-29 21:45:34 +0000770 v9.CheckDefFailure(['def s:Nested()', 'enddef'], 'E1075:')
771 v9.CheckDefFailure(['def b:Nested()', 'enddef'], 'E1075:')
Bram Moolenaar8b848ca2020-09-10 22:28:01 +0200772
Bram Moolenaar54021752020-12-06 18:50:36 +0100773 var lines =<< trim END
774 def Outer()
775 def Inner()
776 # comment
777 enddef
778 def Inner()
779 enddef
780 enddef
781 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000782 v9.CheckDefFailure(lines, 'E1073:')
Bram Moolenaar54021752020-12-06 18:50:36 +0100783
784 lines =<< trim END
785 def Outer()
786 def Inner()
787 # comment
788 enddef
789 def! Inner()
790 enddef
791 enddef
792 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000793 v9.CheckDefFailure(lines, 'E1117:')
Bram Moolenaar54021752020-12-06 18:50:36 +0100794
Bram Moolenaardb8e5c22021-12-25 19:58:22 +0000795 lines =<< trim END
796 vim9script
797 def Outer()
798 def Inner()
799 g:result = 'ok'
800 enddef
801 Inner()
802 enddef
803 Outer()
804 Inner()
805 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000806 v9.CheckScriptFailure(lines, 'E117: Unknown function: Inner')
Bram Moolenaardb8e5c22021-12-25 19:58:22 +0000807 assert_equal('ok', g:result)
808 unlet g:result
809
Bram Moolenaarf681cfb2022-02-07 20:30:57 +0000810 lines =<< trim END
811 vim9script
812 def Outer()
813 def _Inner()
814 echo 'bad'
815 enddef
Bram Moolenaar3787f262022-02-07 21:54:01 +0000816 _Inner()
Bram Moolenaarf681cfb2022-02-07 20:30:57 +0000817 enddef
818 defcompile
819 END
Bram Moolenaar3787f262022-02-07 21:54:01 +0000820 v9.CheckScriptFailure(lines, 'E1267:')
Bram Moolenaarf681cfb2022-02-07 20:30:57 +0000821
822 lines =<< trim END
823 vim9script
824 def Outer()
825 def g:inner()
826 echo 'bad'
827 enddef
Bram Moolenaar3787f262022-02-07 21:54:01 +0000828 g:inner()
Bram Moolenaarf681cfb2022-02-07 20:30:57 +0000829 enddef
830 defcompile
831 END
Bram Moolenaar3787f262022-02-07 21:54:01 +0000832 v9.CheckScriptFailure(lines, 'E1267:')
833
834 lines =<< trim END
835 vim9script
836 def g:_Func()
837 echo 'bad'
838 enddef
839 END
840 v9.CheckScriptFailure(lines, 'E1267:')
841
842 lines =<< trim END
843 vim9script
Bram Moolenaara749a422022-02-12 19:52:25 +0000844 def _Func()
Bram Moolenaar3787f262022-02-07 21:54:01 +0000845 echo 'bad'
846 enddef
847 END
848 v9.CheckScriptFailure(lines, 'E1267:')
Bram Moolenaarf681cfb2022-02-07 20:30:57 +0000849
Bram Moolenaar54021752020-12-06 18:50:36 +0100850 # nested function inside conditional
Bram Moolenaar54021752020-12-06 18:50:36 +0100851 lines =<< trim END
852 vim9script
853 var thecount = 0
854 if true
855 def Test(): number
856 def TheFunc(): number
857 thecount += 1
858 return thecount
859 enddef
860 return TheFunc()
861 enddef
862 endif
863 defcompile
864 assert_equal(1, Test())
865 assert_equal(2, Test())
866 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000867 v9.CheckScriptSuccess(lines)
Bram Moolenaar8863bda2021-03-17 18:42:08 +0100868
869 # also works when "thecount" is inside the "if" block
870 lines =<< trim END
871 vim9script
872 if true
873 var thecount = 0
874 def Test(): number
875 def TheFunc(): number
876 thecount += 1
877 return thecount
878 enddef
879 return TheFunc()
880 enddef
881 endif
882 defcompile
883 assert_equal(1, Test())
884 assert_equal(2, Test())
885 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000886 v9.CheckScriptSuccess(lines)
Bram Moolenaar4bba16d2021-08-15 19:28:05 +0200887
Bram Moolenaara915fa02022-03-23 11:29:15 +0000888 # nested function with recursive call
889 lines =<< trim END
890 vim9script
891
892 def MyFunc(): number
893 def Fib(n: number): number
894 if n < 2
895 return 1
896 endif
897 return Fib(n - 2) + Fib(n - 1)
898 enddef
899
900 return Fib(5)
901 enddef
902
903 assert_equal(8, MyFunc())
904 END
905 v9.CheckScriptSuccess(lines)
906
Bram Moolenaar4bba16d2021-08-15 19:28:05 +0200907 lines =<< trim END
908 vim9script
909 def Outer()
910 def Inner()
911 echo 'hello'
912 enddef burp
913 enddef
914 defcompile
915 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000916 v9.CheckScriptFailure(lines, 'E1173: Text found after enddef: burp', 3)
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200917enddef
918
Bram Moolenaar1889f492022-08-16 19:34:44 +0100919def Test_nested_function_fails()
920 var lines =<< trim END
921 def T()
922 def Func(g: string):string
923 enddef
924 Func()
925 enddef
926 silent! defcompile
927 END
928 v9.CheckScriptFailure(lines, 'E1069:')
929enddef
930
Bram Moolenaaradc8e442020-12-31 18:28:18 +0100931def Test_not_nested_function()
932 echo printf('%d',
933 function('len')('xxx'))
934enddef
935
Bram Moolenaaraf8edbb2020-08-01 00:03:09 +0200936func Test_call_default_args_from_func()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +0200937 call MyDefaultArgs()->assert_equal('string')
938 call MyDefaultArgs('one')->assert_equal('one')
Bram Moolenaar9bd5d872020-09-06 21:47:48 +0200939 call assert_fails('call MyDefaultArgs("one", "two")', 'E118:', '', 3, 'Test_call_default_args_from_func')
Bram Moolenaaraf8edbb2020-08-01 00:03:09 +0200940endfunc
941
Bram Moolenaar38ddf332020-07-31 22:05:04 +0200942def Test_nested_global_function()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +0200943 var lines =<< trim END
Bram Moolenaar38ddf332020-07-31 22:05:04 +0200944 vim9script
945 def Outer()
946 def g:Inner(): string
947 return 'inner'
948 enddef
949 enddef
Bram Moolenaaraf8edbb2020-08-01 00:03:09 +0200950 defcompile
951 Outer()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +0200952 g:Inner()->assert_equal('inner')
Bram Moolenaaraf8edbb2020-08-01 00:03:09 +0200953 delfunc g:Inner
954 Outer()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +0200955 g:Inner()->assert_equal('inner')
Bram Moolenaaraf8edbb2020-08-01 00:03:09 +0200956 delfunc g:Inner
957 Outer()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +0200958 g:Inner()->assert_equal('inner')
Bram Moolenaaraf8edbb2020-08-01 00:03:09 +0200959 delfunc g:Inner
Bram Moolenaar38ddf332020-07-31 22:05:04 +0200960 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000961 v9.CheckScriptSuccess(lines)
Bram Moolenaar2c79e9d2020-08-01 18:57:52 +0200962
963 lines =<< trim END
964 vim9script
965 def Outer()
Bram Moolenaar38453522021-11-28 22:00:12 +0000966 func g:Inner()
967 return 'inner'
968 endfunc
969 enddef
970 defcompile
971 Outer()
972 g:Inner()->assert_equal('inner')
973 delfunc g:Inner
974 Outer()
975 g:Inner()->assert_equal('inner')
976 delfunc g:Inner
977 Outer()
978 g:Inner()->assert_equal('inner')
979 delfunc g:Inner
980 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000981 v9.CheckScriptSuccess(lines)
Bram Moolenaar38453522021-11-28 22:00:12 +0000982
983 lines =<< trim END
984 vim9script
985 def Outer()
Bram Moolenaar2c79e9d2020-08-01 18:57:52 +0200986 def g:Inner(): string
987 return 'inner'
988 enddef
989 enddef
990 defcompile
991 Outer()
992 Outer()
993 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000994 v9.CheckScriptFailure(lines, "E122:")
Bram Moolenaarcd45ed02020-12-22 17:35:54 +0100995 delfunc g:Inner
Bram Moolenaarad486a02020-08-01 23:22:18 +0200996
997 lines =<< trim END
998 vim9script
Bram Moolenaar58a52f22020-12-22 18:56:55 +0100999 def Outer()
1000 def g:Inner()
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01001001 echo map([1, 2, 3], (_, v) => v + 1)
Bram Moolenaar58a52f22020-12-22 18:56:55 +01001002 enddef
1003 g:Inner()
1004 enddef
1005 Outer()
1006 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001007 v9.CheckScriptSuccess(lines)
Bram Moolenaar58a52f22020-12-22 18:56:55 +01001008 delfunc g:Inner
1009
1010 lines =<< trim END
1011 vim9script
Bram Moolenaarad486a02020-08-01 23:22:18 +02001012 def Func()
1013 echo 'script'
1014 enddef
1015 def Outer()
1016 def Func()
1017 echo 'inner'
1018 enddef
1019 enddef
1020 defcompile
1021 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001022 v9.CheckScriptFailure(lines, "E1073:", 1)
Bram Moolenaard604d782021-11-20 21:46:20 +00001023
1024 lines =<< trim END
1025 vim9script
1026 def Func()
1027 echo 'script'
1028 enddef
1029 def Func()
1030 echo 'script'
1031 enddef
1032 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001033 v9.CheckScriptFailure(lines, "E1073:", 5)
Bram Moolenaar38ddf332020-07-31 22:05:04 +02001034enddef
1035
Bram Moolenaar6abdcf82020-11-22 18:15:44 +01001036def DefListAll()
1037 def
1038enddef
1039
1040def DefListOne()
1041 def DefListOne
1042enddef
1043
1044def DefListMatches()
1045 def /DefList
1046enddef
1047
1048def Test_nested_def_list()
1049 var funcs = split(execute('call DefListAll()'), "\n")
1050 assert_true(len(funcs) > 10)
1051 assert_true(funcs->index('def DefListAll()') >= 0)
1052
1053 funcs = split(execute('call DefListOne()'), "\n")
1054 assert_equal([' def DefListOne()', '1 def DefListOne', ' enddef'], funcs)
1055
1056 funcs = split(execute('call DefListMatches()'), "\n")
1057 assert_true(len(funcs) >= 3)
1058 assert_true(funcs->index('def DefListAll()') >= 0)
1059 assert_true(funcs->index('def DefListOne()') >= 0)
1060 assert_true(funcs->index('def DefListMatches()') >= 0)
Bram Moolenaar54021752020-12-06 18:50:36 +01001061
1062 var lines =<< trim END
1063 vim9script
1064 def Func()
1065 def +Func+
1066 enddef
1067 defcompile
1068 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001069 v9.CheckScriptFailure(lines, 'E476:', 1)
Bram Moolenaar6abdcf82020-11-22 18:15:44 +01001070enddef
1071
Bram Moolenaare08be092022-02-17 13:08:26 +00001072def Test_global_function_not_found()
1073 var lines =<< trim END
1074 g:Ref = 123
1075 call g:Ref()
1076 END
1077 v9.CheckDefExecAndScriptFailure(lines, ['E117:', 'E1085:'], 2)
1078enddef
1079
Bram Moolenaar333894b2020-08-01 18:53:07 +02001080def Test_global_local_function()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001081 var lines =<< trim END
Bram Moolenaar333894b2020-08-01 18:53:07 +02001082 vim9script
1083 def g:Func(): string
1084 return 'global'
1085 enddef
1086 def Func(): string
1087 return 'local'
1088 enddef
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001089 g:Func()->assert_equal('global')
1090 Func()->assert_equal('local')
Bram Moolenaar2d870f82020-12-05 13:41:01 +01001091 delfunc g:Func
Bram Moolenaar333894b2020-08-01 18:53:07 +02001092 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001093 v9.CheckScriptSuccess(lines)
Bram Moolenaar035d6e92020-08-11 22:30:42 +02001094
1095 lines =<< trim END
1096 vim9script
1097 def g:Funcy()
1098 echo 'funcy'
1099 enddef
Bram Moolenaara749a422022-02-12 19:52:25 +00001100 Funcy()
Bram Moolenaar035d6e92020-08-11 22:30:42 +02001101 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001102 v9.CheckScriptFailure(lines, 'E117:')
Bram Moolenaar333894b2020-08-01 18:53:07 +02001103enddef
1104
Bram Moolenaar0f769812020-09-12 18:32:34 +02001105def Test_local_function_shadows_global()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001106 var lines =<< trim END
Bram Moolenaar0f769812020-09-12 18:32:34 +02001107 vim9script
1108 def g:Gfunc(): string
1109 return 'global'
1110 enddef
1111 def AnotherFunc(): number
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001112 var Gfunc = function('len')
Bram Moolenaar0f769812020-09-12 18:32:34 +02001113 return Gfunc('testing')
1114 enddef
1115 g:Gfunc()->assert_equal('global')
1116 AnotherFunc()->assert_equal(7)
1117 delfunc g:Gfunc
1118 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001119 v9.CheckScriptSuccess(lines)
Bram Moolenaar0f769812020-09-12 18:32:34 +02001120
1121 lines =<< trim END
1122 vim9script
1123 def g:Func(): string
1124 return 'global'
1125 enddef
1126 def AnotherFunc()
1127 g:Func = function('len')
1128 enddef
1129 AnotherFunc()
1130 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001131 v9.CheckScriptFailure(lines, 'E705:')
Bram Moolenaar0f769812020-09-12 18:32:34 +02001132 delfunc g:Func
Bram Moolenaar0865b152021-04-05 15:38:51 +02001133
Bram Moolenaar62aec932022-01-29 21:45:34 +00001134 # global function is not found with g: prefix
Bram Moolenaar0865b152021-04-05 15:38:51 +02001135 lines =<< trim END
1136 vim9script
1137 def g:Func(): string
1138 return 'global'
1139 enddef
1140 def AnotherFunc(): string
1141 return Func()
1142 enddef
1143 assert_equal('global', AnotherFunc())
Bram Moolenaar0865b152021-04-05 15:38:51 +02001144 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001145 v9.CheckScriptFailure(lines, 'E117:')
1146 delfunc g:Func
Bram Moolenaar0865b152021-04-05 15:38:51 +02001147
1148 lines =<< trim END
1149 vim9script
1150 def g:Func(): string
1151 return 'global'
1152 enddef
Bram Moolenaar848fadd2022-01-30 15:28:30 +00001153 assert_equal('global', g:Func())
Bram Moolenaar0865b152021-04-05 15:38:51 +02001154 delfunc g:Func
1155 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001156 v9.CheckScriptSuccess(lines)
Bram Moolenaar58493cf2022-01-06 12:23:30 +00001157
1158 # This does not shadow "i" which is visible only inside the for loop
1159 lines =<< trim END
1160 vim9script
1161
1162 def Foo(i: number)
1163 echo i
1164 enddef
1165
1166 for i in range(3)
1167 # Foo() is compiled here
1168 Foo(i)
1169 endfor
1170 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001171 v9.CheckScriptSuccess(lines)
Bram Moolenaar0f769812020-09-12 18:32:34 +02001172enddef
1173
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001174func TakesOneArg(arg)
1175 echo a:arg
1176endfunc
1177
1178def Test_call_wrong_args()
Bram Moolenaar62aec932022-01-29 21:45:34 +00001179 v9.CheckDefFailure(['g:TakesOneArg()'], 'E119:')
1180 v9.CheckDefFailure(['g:TakesOneArg(11, 22)'], 'E118:')
1181 v9.CheckDefFailure(['bufnr(xxx)'], 'E1001:')
1182 v9.CheckScriptFailure(['def Func(Ref: func(s: string))'], 'E475:')
Bram Moolenaaree8580e2020-08-28 17:19:07 +02001183
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001184 var lines =<< trim END
Bram Moolenaaree8580e2020-08-28 17:19:07 +02001185 vim9script
1186 def Func(s: string)
1187 echo s
1188 enddef
1189 Func([])
1190 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001191 v9.CheckScriptFailure(lines, 'E1013: Argument 1: type mismatch, expected string but got list<unknown>', 5)
Bram Moolenaarb185a402020-09-18 22:42:00 +02001192
Bram Moolenaar9a015112021-12-31 14:06:45 +00001193 # argument name declared earlier is found when declaring a function
Bram Moolenaarb185a402020-09-18 22:42:00 +02001194 lines =<< trim END
1195 vim9script
Bram Moolenaarb4893b82021-02-21 22:20:24 +01001196 var name = 'piet'
1197 def FuncOne(name: string)
Bram Moolenaar3a5988c2022-02-08 19:23:35 +00001198 echo name
Bram Moolenaarb4893b82021-02-21 22:20:24 +01001199 enddef
1200 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001201 v9.CheckScriptFailure(lines, 'E1168:')
Bram Moolenaarb4893b82021-02-21 22:20:24 +01001202
Bram Moolenaar3a5988c2022-02-08 19:23:35 +00001203 # same, inside the same block
1204 lines =<< trim END
1205 vim9script
1206 if true
1207 var name = 'piet'
1208 def FuncOne(name: string)
1209 echo name
1210 enddef
1211 endif
1212 END
1213 v9.CheckScriptFailure(lines, 'E1168:')
1214
1215 # variable in other block is OK
1216 lines =<< trim END
1217 vim9script
1218 if true
1219 var name = 'piet'
1220 endif
1221 def FuncOne(name: string)
1222 echo name
1223 enddef
1224 END
1225 v9.CheckScriptSuccess(lines)
1226
Bram Moolenaardce24412022-02-08 20:35:30 +00001227 # with another variable in another block
1228 lines =<< trim END
1229 vim9script
1230 if true
1231 var name = 'piet'
1232 # define a function so that the variable isn't cleared
1233 def GetItem(): string
1234 return item
1235 enddef
1236 endif
1237 if true
1238 var name = 'peter'
1239 def FuncOne(name: string)
1240 echo name
1241 enddef
1242 endif
1243 END
1244 v9.CheckScriptFailure(lines, 'E1168:')
1245
1246 # only variable in another block is OK
1247 lines =<< trim END
1248 vim9script
1249 if true
1250 var name = 'piet'
1251 # define a function so that the variable isn't cleared
1252 def GetItem(): string
1253 return item
1254 enddef
1255 endif
1256 if true
1257 def FuncOne(name: string)
1258 echo name
1259 enddef
1260 endif
1261 END
1262 v9.CheckScriptSuccess(lines)
1263
Bram Moolenaar9a015112021-12-31 14:06:45 +00001264 # argument name declared later is only found when compiling
1265 lines =<< trim END
1266 vim9script
1267 def FuncOne(name: string)
1268 echo nr
1269 enddef
1270 var name = 'piet'
1271 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001272 v9.CheckScriptSuccess(lines)
1273 v9.CheckScriptFailure(lines + ['defcompile'], 'E1168:')
Bram Moolenaar9a015112021-12-31 14:06:45 +00001274
Bram Moolenaarb4893b82021-02-21 22:20:24 +01001275 lines =<< trim END
1276 vim9script
Bram Moolenaarb185a402020-09-18 22:42:00 +02001277 def FuncOne(nr: number)
1278 echo nr
1279 enddef
1280 def FuncTwo()
1281 FuncOne()
1282 enddef
1283 defcompile
1284 END
1285 writefile(lines, 'Xscript')
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001286 var didCatch = false
Bram Moolenaarb185a402020-09-18 22:42:00 +02001287 try
1288 source Xscript
1289 catch
1290 assert_match('E119: Not enough 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 lines =<< trim END
1297 vim9script
1298 def FuncOne(nr: number)
1299 echo nr
1300 enddef
1301 def FuncTwo()
1302 FuncOne(1, 2)
1303 enddef
1304 defcompile
1305 END
Bram Moolenaarf5fec052022-09-11 11:49:22 +01001306 writefile(lines, 'Xscript', 'D')
Bram Moolenaarb185a402020-09-18 22:42:00 +02001307 didCatch = false
1308 try
1309 source Xscript
1310 catch
1311 assert_match('E118: Too many arguments for function: <SNR>\d\+_FuncOne', v:exception)
1312 assert_match('Xscript\[8\]..function <SNR>\d\+_FuncTwo, line 1', v:throwpoint)
1313 didCatch = true
1314 endtry
1315 assert_true(didCatch)
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001316enddef
1317
Bram Moolenaar50824712020-12-20 21:10:17 +01001318def Test_call_funcref_wrong_args()
1319 var head =<< trim END
1320 vim9script
1321 def Func3(a1: string, a2: number, a3: list<number>)
1322 echo a1 .. a2 .. a3[0]
1323 enddef
1324 def Testme()
1325 var funcMap: dict<func> = {func: Func3}
1326 END
1327 var tail =<< trim END
1328 enddef
1329 Testme()
1330 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001331 v9.CheckScriptSuccess(head + ["funcMap['func']('str', 123, [1, 2, 3])"] + tail)
Bram Moolenaar50824712020-12-20 21:10:17 +01001332
Bram Moolenaar62aec932022-01-29 21:45:34 +00001333 v9.CheckScriptFailure(head + ["funcMap['func']('str', 123)"] + tail, 'E119:')
1334 v9.CheckScriptFailure(head + ["funcMap['func']('str', 123, [1], 4)"] + tail, 'E118:')
Bram Moolenaar32b3f822021-01-06 21:59:39 +01001335
1336 var lines =<< trim END
1337 vim9script
1338 var Ref: func(number): any
1339 Ref = (j) => !j
1340 echo Ref(false)
1341 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001342 v9.CheckScriptFailure(lines, 'E1013: Argument 1: type mismatch, expected number but got bool', 4)
Bram Moolenaar32b3f822021-01-06 21:59:39 +01001343
1344 lines =<< trim END
1345 vim9script
1346 var Ref: func(number): any
1347 Ref = (j) => !j
1348 call Ref(false)
1349 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001350 v9.CheckScriptFailure(lines, 'E1013: Argument 1: type mismatch, expected number but got bool', 4)
Bram Moolenaar50824712020-12-20 21:10:17 +01001351enddef
1352
Bram Moolenaarb4d16cb2020-11-05 18:45:46 +01001353def Test_call_lambda_args()
Bram Moolenaar2a389082021-04-09 20:24:31 +02001354 var lines =<< trim END
1355 var Callback = (..._) => 'anything'
1356 assert_equal('anything', Callback())
1357 assert_equal('anything', Callback(1))
1358 assert_equal('anything', Callback('a', 2))
Bram Moolenaar1088b692021-04-09 22:12:44 +02001359
1360 assert_equal('xyz', ((a: string): string => a)('xyz'))
Bram Moolenaar2a389082021-04-09 20:24:31 +02001361 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001362 v9.CheckDefAndScriptSuccess(lines)
Bram Moolenaar2a389082021-04-09 20:24:31 +02001363
Bram Moolenaar62aec932022-01-29 21:45:34 +00001364 v9.CheckDefFailure(['echo ((i) => 0)()'],
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01001365 'E119: Not enough arguments for function: ((i) => 0)()')
Bram Moolenaarb4d16cb2020-11-05 18:45:46 +01001366
Bram Moolenaar2a389082021-04-09 20:24:31 +02001367 lines =<< trim END
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01001368 var Ref = (x: number, y: number) => x + y
Bram Moolenaarb4d16cb2020-11-05 18:45:46 +01001369 echo Ref(1, 'x')
1370 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001371 v9.CheckDefFailure(lines, 'E1013: Argument 2: type mismatch, expected number but got string')
Bram Moolenaare68b02a2021-01-03 13:09:51 +01001372
1373 lines =<< trim END
1374 var Ref: func(job, string, number)
1375 Ref = (x, y) => 0
1376 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001377 v9.CheckDefAndScriptFailure(lines, 'E1012:')
Bram Moolenaare68b02a2021-01-03 13:09:51 +01001378
1379 lines =<< trim END
1380 var Ref: func(job, string)
1381 Ref = (x, y, z) => 0
1382 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001383 v9.CheckDefAndScriptFailure(lines, 'E1012:')
Bram Moolenaar057e84a2021-02-28 16:55:11 +01001384
1385 lines =<< trim END
1386 var one = 1
1387 var l = [1, 2, 3]
1388 echo map(l, (one) => one)
1389 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001390 v9.CheckDefFailure(lines, 'E1167:')
1391 v9.CheckScriptFailure(['vim9script'] + lines, 'E1168:')
Bram Moolenaar057e84a2021-02-28 16:55:11 +01001392
1393 lines =<< trim END
Bram Moolenaar14ded112021-06-26 19:25:49 +02001394 var Ref: func(any, ?any): bool
1395 Ref = (_, y = 1) => false
1396 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001397 v9.CheckDefAndScriptFailure(lines, 'E1172:')
Bram Moolenaar14ded112021-06-26 19:25:49 +02001398
1399 lines =<< trim END
Bram Moolenaar015cf102021-06-26 21:52:02 +02001400 var a = 0
1401 var b = (a == 0 ? 1 : 2)
1402 assert_equal(1, b)
Bram Moolenaar98f9a5f2021-06-26 22:22:38 +02001403 var txt = 'a'
1404 b = (txt =~ 'x' ? 1 : 2)
1405 assert_equal(2, b)
Bram Moolenaar015cf102021-06-26 21:52:02 +02001406 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001407 v9.CheckDefAndScriptSuccess(lines)
Bram Moolenaar015cf102021-06-26 21:52:02 +02001408
1409 lines =<< trim END
Bram Moolenaar057e84a2021-02-28 16:55:11 +01001410 def ShadowLocal()
1411 var one = 1
1412 var l = [1, 2, 3]
1413 echo map(l, (one) => one)
1414 enddef
1415 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001416 v9.CheckDefFailure(lines, 'E1167:')
Bram Moolenaar057e84a2021-02-28 16:55:11 +01001417
1418 lines =<< trim END
1419 def Shadowarg(one: number)
1420 var l = [1, 2, 3]
1421 echo map(l, (one) => one)
1422 enddef
1423 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001424 v9.CheckDefFailure(lines, 'E1167:')
Bram Moolenaar767034c2021-04-09 17:24:52 +02001425
1426 lines =<< trim END
1427 echo ((a) => a)('aa', 'bb')
1428 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001429 v9.CheckDefAndScriptFailure(lines, 'E118:', 1)
Bram Moolenaarc4c56422021-07-21 20:38:46 +02001430
1431 lines =<< trim END
1432 echo 'aa'->((a) => a)('bb')
1433 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001434 v9.CheckDefFailure(lines, 'E118: Too many arguments for function: ->((a) => a)(''bb'')', 1)
1435 v9.CheckScriptFailure(['vim9script'] + lines, 'E118: Too many arguments for function: <lambda>', 2)
Bram Moolenaarb4d16cb2020-11-05 18:45:46 +01001436enddef
1437
Bram Moolenaara755fdb2021-11-20 21:35:41 +00001438def Test_lambda_line_nr()
1439 var lines =<< trim END
1440 vim9script
1441 # comment
1442 # comment
1443 var id = timer_start(1'000, (_) => 0)
1444 var out = execute('verbose ' .. timer_info(id)[0].callback
1445 ->string()
1446 ->substitute("('\\|')", ' ', 'g'))
1447 assert_match('Last set from .* line 4', out)
1448 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001449 v9.CheckScriptSuccess(lines)
Bram Moolenaara755fdb2021-11-20 21:35:41 +00001450enddef
1451
Bram Moolenaar5f91e742021-03-17 21:29:29 +01001452def FilterWithCond(x: string, Cond: func(string): bool): bool
1453 return Cond(x)
1454enddef
1455
Bram Moolenaar0346b792021-01-31 22:18:29 +01001456def Test_lambda_return_type()
1457 var lines =<< trim END
1458 var Ref = (): => 123
1459 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001460 v9.CheckDefAndScriptFailure(lines, 'E1157:', 1)
Bram Moolenaar5f91e742021-03-17 21:29:29 +01001461
Yegappan Lakshmanan611728f2021-05-24 15:15:47 +02001462 # no space before the return type
1463 lines =<< trim END
1464 var Ref = (x):number => x + 1
1465 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001466 v9.CheckDefAndScriptFailure(lines, 'E1069:', 1)
Yegappan Lakshmanan611728f2021-05-24 15:15:47 +02001467
Bram Moolenaar5f91e742021-03-17 21:29:29 +01001468 # this works
1469 for x in ['foo', 'boo']
Bram Moolenaar62aec932022-01-29 21:45:34 +00001470 echo g:FilterWithCond(x, (v) => v =~ '^b')
Bram Moolenaar5f91e742021-03-17 21:29:29 +01001471 endfor
1472
1473 # this fails
1474 lines =<< trim END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001475 echo g:FilterWithCond('foo', (v) => v .. '^b')
Bram Moolenaar5f91e742021-03-17 21:29:29 +01001476 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001477 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 +02001478
1479 lines =<< trim END
1480 var Lambda1 = (x) => {
1481 return x
1482 }
1483 assert_equal('asdf', Lambda1('asdf'))
1484 var Lambda2 = (x): string => {
1485 return x
1486 }
1487 assert_equal('foo', Lambda2('foo'))
1488 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001489 v9.CheckDefAndScriptSuccess(lines)
Bram Moolenaara9931532021-06-12 15:58:16 +02001490
1491 lines =<< trim END
1492 var Lambda = (x): string => {
1493 return x
1494 }
1495 echo Lambda(['foo'])
1496 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001497 v9.CheckDefExecAndScriptFailure(lines, 'E1012:')
Bram Moolenaar0346b792021-01-31 22:18:29 +01001498enddef
1499
Bram Moolenaar709664c2020-12-12 14:33:41 +01001500def Test_lambda_uses_assigned_var()
Bram Moolenaar62aec932022-01-29 21:45:34 +00001501 v9.CheckDefSuccess([
Bram Moolenaar2984ed32022-08-20 14:51:17 +01001502 'var x: any = "aaa"',
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01001503 'x = filter(["bbb"], (_, v) => v =~ x)'])
Bram Moolenaar709664c2020-12-12 14:33:41 +01001504enddef
1505
Bram Moolenaarf5f4e852022-09-22 22:03:14 +01001506def Test_lambda_invalid_block()
1507 var lines =<< trim END
1508 timer_start(0, (_) => { # echo
1509 echo 'yes'
1510 })
1511 END
1512 v9.CheckDefAndScriptSuccess(lines)
1513
1514 lines =<< trim END
1515 timer_start(0, (_) => { " echo
1516 echo 'yes'
1517 })
1518 END
1519 v9.CheckDefAndScriptFailure(lines, 'E488: Trailing characters: " echo')
1520
1521 lines =<< trim END
1522 timer_start(0, (_) => { | echo
1523 echo 'yes'
1524 })
1525 END
1526 v9.CheckDefAndScriptFailure(lines, 'E488: Trailing characters: | echo')
1527enddef
1528
Bram Moolenaarf8addf12022-09-23 12:44:25 +01001529def Test_lambda_with_following_cmd()
1530 var lines =<< trim END
1531 set ts=2
1532 var Lambda = () => {
1533 set ts=4
1534 } | set ts=3
1535 assert_equal(3, &ts)
1536 Lambda()
1537 assert_equal(4, &ts)
1538 END
1539 v9.CheckDefAndScriptSuccess(lines)
1540 set ts=8
1541enddef
1542
Bram Moolenaar18062fc2021-03-05 21:35:47 +01001543def Test_pass_legacy_lambda_to_def_func()
1544 var lines =<< trim END
1545 vim9script
1546 func Foo()
1547 eval s:Bar({x -> 0})
1548 endfunc
1549 def Bar(y: any)
1550 enddef
1551 Foo()
1552 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001553 v9.CheckScriptSuccess(lines)
Bram Moolenaar831bdf82021-06-22 19:32:17 +02001554
1555 lines =<< trim END
1556 vim9script
Bram Moolenaar1d9cef72022-03-17 16:30:03 +00001557 def g:TestFunc(F: func)
Bram Moolenaar831bdf82021-06-22 19:32:17 +02001558 enddef
1559 legacy call g:TestFunc({-> 0})
1560 delfunc g:TestFunc
1561
Bram Moolenaar1d9cef72022-03-17 16:30:03 +00001562 def g:TestFunc(F: func(number))
Bram Moolenaar831bdf82021-06-22 19:32:17 +02001563 enddef
1564 legacy call g:TestFunc({nr -> 0})
1565 delfunc g:TestFunc
1566 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001567 v9.CheckScriptSuccess(lines)
Bram Moolenaar18062fc2021-03-05 21:35:47 +01001568enddef
1569
Bram Moolenaar844fb642021-10-23 13:32:30 +01001570def Test_lambda_in_reduce_line_break()
1571 # this was using freed memory
1572 var lines =<< trim END
1573 vim9script
1574 const result: dict<number> =
1575 ['Bob', 'Sam', 'Cat', 'Bob', 'Cat', 'Cat']
1576 ->reduce((acc, val) => {
1577 if has_key(acc, val)
1578 acc[val] += 1
1579 return acc
1580 else
1581 acc[val] = 1
1582 return acc
1583 endif
1584 }, {})
1585 assert_equal({Bob: 2, Sam: 1, Cat: 3}, result)
1586 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001587 v9.CheckScriptSuccess(lines)
Bram Moolenaar844fb642021-10-23 13:32:30 +01001588enddef
1589
Bram Moolenaardcb53be2021-12-09 14:23:43 +00001590def Test_set_opfunc_to_lambda()
1591 var lines =<< trim END
1592 vim9script
1593 nnoremap <expr> <F4> <SID>CountSpaces() .. '_'
1594 def CountSpaces(type = ''): string
1595 if type == ''
1596 &operatorfunc = (t) => CountSpaces(t)
1597 return 'g@'
1598 endif
1599 normal! '[V']y
1600 g:result = getreg('"')->count(' ')
1601 return ''
1602 enddef
1603 new
1604 'a b c d e'->setline(1)
1605 feedkeys("\<F4>", 'x')
1606 assert_equal(4, g:result)
1607 bwipe!
1608 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001609 v9.CheckScriptSuccess(lines)
Bram Moolenaardcb53be2021-12-09 14:23:43 +00001610enddef
1611
Bram Moolenaaref082e12021-12-12 21:02:03 +00001612def Test_set_opfunc_to_global_function()
1613 var lines =<< trim END
1614 vim9script
1615 def g:CountSpaces(type = ''): string
1616 normal! '[V']y
1617 g:result = getreg('"')->count(' ')
1618 return ''
1619 enddef
Bram Moolenaarb15cf442021-12-16 15:49:43 +00001620 # global function works at script level
Bram Moolenaaref082e12021-12-12 21:02:03 +00001621 &operatorfunc = g:CountSpaces
1622 new
1623 'a b c d e'->setline(1)
1624 feedkeys("g@_", 'x')
1625 assert_equal(4, g:result)
Bram Moolenaarb15cf442021-12-16 15:49:43 +00001626
1627 &operatorfunc = ''
1628 g:result = 0
1629 # global function works in :def function
1630 def Func()
1631 &operatorfunc = g:CountSpaces
1632 enddef
1633 Func()
1634 feedkeys("g@_", 'x')
1635 assert_equal(4, g:result)
1636
Bram Moolenaaref082e12021-12-12 21:02:03 +00001637 bwipe!
1638 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001639 v9.CheckScriptSuccess(lines)
Bram Moolenaaref082e12021-12-12 21:02:03 +00001640 &operatorfunc = ''
1641enddef
1642
Bram Moolenaar33b968d2021-12-13 11:31:04 +00001643def Test_use_script_func_name_with_prefix()
1644 var lines =<< trim END
1645 vim9script
Bram Moolenaara749a422022-02-12 19:52:25 +00001646 func g:Getit()
Bram Moolenaar33b968d2021-12-13 11:31:04 +00001647 return 'it'
1648 endfunc
Bram Moolenaara749a422022-02-12 19:52:25 +00001649 var Fn = g:Getit
Bram Moolenaar33b968d2021-12-13 11:31:04 +00001650 assert_equal('it', Fn())
1651 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001652 v9.CheckScriptSuccess(lines)
Bram Moolenaar33b968d2021-12-13 11:31:04 +00001653enddef
1654
Bram Moolenaardd297bc2021-12-10 10:37:38 +00001655def Test_lambda_type_allocated()
1656 # Check that unreferencing a partial using a lambda can use the variable type
1657 # after the lambda has been freed and does not leak memory.
1658 var lines =<< trim END
1659 vim9script
1660
1661 func MyomniFunc1(val, findstart, base)
1662 return a:findstart ? 0 : []
1663 endfunc
1664
1665 var Lambda = (a, b) => MyomniFunc1(19, a, b)
1666 &omnifunc = Lambda
1667 Lambda = (a, b) => MyomniFunc1(20, a, b)
1668 &omnifunc = string(Lambda)
1669 Lambda = (a, b) => strlen(a)
1670 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001671 v9.CheckScriptSuccess(lines)
Bram Moolenaardd297bc2021-12-10 10:37:38 +00001672enddef
1673
Bram Moolenaara7583c42022-05-07 21:14:05 +01001674def Test_define_lambda_in_execute()
1675 var lines =<< trim [CODE]
1676 vim9script
1677
1678 def BuildFuncMultiLine(): func
1679 var x =<< trim END
1680 g:SomeRandomFunc = (d: dict<any>) => {
1681 return d.k1 + d.k2
1682 }
1683 END
1684 execute(x)
1685 return g:SomeRandomFunc
1686 enddef
1687 var ResultPlus = BuildFuncMultiLine()
1688 assert_equal(7, ResultPlus({k1: 3, k2: 4}))
1689 [CODE]
1690 v9.CheckScriptSuccess(lines)
1691 unlet g:SomeRandomFunc
1692enddef
1693
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001694" Default arg and varargs
1695def MyDefVarargs(one: string, two = 'foo', ...rest: list<string>): string
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001696 var res = one .. ',' .. two
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001697 for s in rest
1698 res ..= ',' .. s
1699 endfor
1700 return res
1701enddef
1702
1703def Test_call_def_varargs()
Bram Moolenaar62aec932022-01-29 21:45:34 +00001704 assert_fails('g:MyDefVarargs()', 'E119:', '', 1, 'Test_call_def_varargs')
1705 g:MyDefVarargs('one')->assert_equal('one,foo')
1706 g:MyDefVarargs('one', 'two')->assert_equal('one,two')
1707 g:MyDefVarargs('one', 'two', 'three')->assert_equal('one,two,three')
1708 v9.CheckDefFailure(['g:MyDefVarargs("one", 22)'],
Bram Moolenaar77072282020-09-16 17:55:40 +02001709 'E1013: Argument 2: type mismatch, expected string but got number')
Bram Moolenaar62aec932022-01-29 21:45:34 +00001710 v9.CheckDefFailure(['g:MyDefVarargs("one", "two", 123)'],
Bram Moolenaar77072282020-09-16 17:55:40 +02001711 'E1013: Argument 3: type mismatch, expected string but got number')
Bram Moolenaar24aa48b2020-07-25 16:33:02 +02001712
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001713 var lines =<< trim END
Bram Moolenaar24aa48b2020-07-25 16:33:02 +02001714 vim9script
1715 def Func(...l: list<string>)
1716 echo l
1717 enddef
1718 Func('a', 'b', 'c')
1719 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001720 v9.CheckScriptSuccess(lines)
Bram Moolenaar24aa48b2020-07-25 16:33:02 +02001721
1722 lines =<< trim END
1723 vim9script
1724 def Func(...l: list<string>)
1725 echo l
1726 enddef
1727 Func()
1728 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001729 v9.CheckScriptSuccess(lines)
Bram Moolenaar24aa48b2020-07-25 16:33:02 +02001730
1731 lines =<< trim END
1732 vim9script
Bram Moolenaar2a389082021-04-09 20:24:31 +02001733 def Func(...l: list<any>)
Bram Moolenaar2f8cbc42020-09-16 17:22:59 +02001734 echo l
1735 enddef
1736 Func(0)
1737 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001738 v9.CheckScriptSuccess(lines)
Bram Moolenaar2f8cbc42020-09-16 17:22:59 +02001739
1740 lines =<< trim END
1741 vim9script
Bram Moolenaar2a389082021-04-09 20:24:31 +02001742 def Func(...l: any)
1743 echo l
1744 enddef
1745 Func(0)
1746 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001747 v9.CheckScriptFailure(lines, 'E1180:', 2)
Bram Moolenaar2a389082021-04-09 20:24:31 +02001748
1749 lines =<< trim END
1750 vim9script
Bram Moolenaar28022722020-09-21 22:02:49 +02001751 def Func(..._l: list<string>)
1752 echo _l
1753 enddef
1754 Func('a', 'b', 'c')
1755 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001756 v9.CheckScriptSuccess(lines)
Bram Moolenaar28022722020-09-21 22:02:49 +02001757
1758 lines =<< trim END
1759 vim9script
Bram Moolenaar24aa48b2020-07-25 16:33:02 +02001760 def Func(...l: list<string>)
1761 echo l
1762 enddef
1763 Func(1, 2, 3)
1764 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001765 v9.CheckScriptFailure(lines, 'E1013: Argument 1: type mismatch')
Bram Moolenaar24aa48b2020-07-25 16:33:02 +02001766
1767 lines =<< trim END
1768 vim9script
1769 def Func(...l: list<string>)
1770 echo l
1771 enddef
1772 Func('a', 9)
1773 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001774 v9.CheckScriptFailure(lines, 'E1013: Argument 2: type mismatch')
Bram Moolenaar24aa48b2020-07-25 16:33:02 +02001775
1776 lines =<< trim END
1777 vim9script
1778 def Func(...l: list<string>)
1779 echo l
1780 enddef
1781 Func(1, 'a')
1782 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001783 v9.CheckScriptFailure(lines, 'E1013: Argument 1: type mismatch')
Bram Moolenaar4f53b792021-02-07 15:59:49 +01001784
1785 lines =<< trim END
1786 vim9script
1787 def Func( # some comment
1788 ...l = []
1789 )
1790 echo l
1791 enddef
1792 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001793 v9.CheckScriptFailure(lines, 'E1160:')
Bram Moolenaar6ce46b92021-08-07 15:35:36 +02001794
1795 lines =<< trim END
1796 vim9script
1797 def DoIt()
1798 g:Later('')
1799 enddef
1800 defcompile
1801 def g:Later(...l: list<number>)
1802 enddef
1803 DoIt()
1804 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001805 v9.CheckScriptFailure(lines, 'E1013: Argument 1: type mismatch, expected number but got string')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001806enddef
1807
Bram Moolenaar1378fbc2020-04-11 20:50:33 +02001808let s:value = ''
1809
1810def FuncOneDefArg(opt = 'text')
1811 s:value = opt
1812enddef
1813
1814def FuncTwoDefArg(nr = 123, opt = 'text'): string
1815 return nr .. opt
1816enddef
1817
1818def FuncVarargs(...arg: list<string>): string
1819 return join(arg, ',')
1820enddef
1821
1822def Test_func_type_varargs()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001823 var RefDefArg: func(?string)
Bram Moolenaar848fadd2022-01-30 15:28:30 +00001824 RefDefArg = g:FuncOneDefArg
Bram Moolenaar1378fbc2020-04-11 20:50:33 +02001825 RefDefArg()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001826 s:value->assert_equal('text')
Bram Moolenaar1378fbc2020-04-11 20:50:33 +02001827 RefDefArg('some')
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001828 s:value->assert_equal('some')
Bram Moolenaar1378fbc2020-04-11 20:50:33 +02001829
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001830 var RefDef2Arg: func(?number, ?string): string
Bram Moolenaar848fadd2022-01-30 15:28:30 +00001831 RefDef2Arg = g:FuncTwoDefArg
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001832 RefDef2Arg()->assert_equal('123text')
1833 RefDef2Arg(99)->assert_equal('99text')
1834 RefDef2Arg(77, 'some')->assert_equal('77some')
Bram Moolenaar1378fbc2020-04-11 20:50:33 +02001835
Bram Moolenaar62aec932022-01-29 21:45:34 +00001836 v9.CheckDefFailure(['var RefWrong: func(string?)'], 'E1010:')
1837 v9.CheckDefFailure(['var RefWrong: func(?string, string)'], 'E1007:')
Bram Moolenaar1378fbc2020-04-11 20:50:33 +02001838
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001839 var RefVarargs: func(...list<string>): string
Bram Moolenaar848fadd2022-01-30 15:28:30 +00001840 RefVarargs = g:FuncVarargs
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001841 RefVarargs()->assert_equal('')
1842 RefVarargs('one')->assert_equal('one')
1843 RefVarargs('one', 'two')->assert_equal('one,two')
Bram Moolenaar1378fbc2020-04-11 20:50:33 +02001844
Bram Moolenaar62aec932022-01-29 21:45:34 +00001845 v9.CheckDefFailure(['var RefWrong: func(...list<string>, string)'], 'E110:')
1846 v9.CheckDefFailure(['var RefWrong: func(...list<string>, ?string)'], 'E110:')
Bram Moolenaar1378fbc2020-04-11 20:50:33 +02001847enddef
1848
Bram Moolenaar0b76b422020-04-07 22:05:08 +02001849" Only varargs
1850def MyVarargsOnly(...args: list<string>): string
1851 return join(args, ',')
1852enddef
1853
1854def Test_call_varargs_only()
Bram Moolenaar62aec932022-01-29 21:45:34 +00001855 g:MyVarargsOnly()->assert_equal('')
1856 g:MyVarargsOnly('one')->assert_equal('one')
1857 g:MyVarargsOnly('one', 'two')->assert_equal('one,two')
1858 v9.CheckDefFailure(['g:MyVarargsOnly(1)'], 'E1013: Argument 1: type mismatch, expected string but got number')
1859 v9.CheckDefFailure(['g:MyVarargsOnly("one", 2)'], 'E1013: Argument 2: type mismatch, expected string but got number')
Bram Moolenaar0b76b422020-04-07 22:05:08 +02001860enddef
1861
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001862def Test_using_var_as_arg()
Bram Moolenaard2939812021-12-30 17:09:05 +00001863 var lines =<< trim END
1864 def Func(x: number)
1865 var x = 234
1866 enddef
1867 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001868 v9.CheckDefFailure(lines, 'E1006:')
Bram Moolenaard2939812021-12-30 17:09:05 +00001869
1870 lines =<< trim END
1871 def Func(Ref: number)
1872 def Ref()
1873 enddef
1874 enddef
1875 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001876 v9.CheckDefFailure(lines, 'E1073:')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001877enddef
1878
Bram Moolenaar62aec932022-01-29 21:45:34 +00001879def s:DictArg(arg: dict<string>)
Bram Moolenaarcb2bdb12020-05-10 22:53:56 +02001880 arg['key'] = 'value'
1881enddef
1882
Bram Moolenaar62aec932022-01-29 21:45:34 +00001883def s:ListArg(arg: list<string>)
Bram Moolenaarcb2bdb12020-05-10 22:53:56 +02001884 arg[0] = 'value'
1885enddef
1886
1887def Test_assign_to_argument()
Bram Moolenaarf5be8cd2020-07-17 20:36:00 +02001888 # works for dict and list
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001889 var d: dict<string> = {}
Bram Moolenaarcb2bdb12020-05-10 22:53:56 +02001890 DictArg(d)
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001891 d['key']->assert_equal('value')
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001892 var l: list<string> = []
Bram Moolenaarcb2bdb12020-05-10 22:53:56 +02001893 ListArg(l)
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001894 l[0]->assert_equal('value')
Bram Moolenaarcb2bdb12020-05-10 22:53:56 +02001895
Bram Moolenaar62aec932022-01-29 21:45:34 +00001896 v9.CheckScriptFailure(['def Func(arg: number)', 'arg = 3', 'enddef', 'defcompile'], 'E1090:')
Bram Moolenaar2d870f82020-12-05 13:41:01 +01001897 delfunc! g:Func
Bram Moolenaarcb2bdb12020-05-10 22:53:56 +02001898enddef
1899
Bram Moolenaarb816dae2020-09-20 22:04:00 +02001900" These argument names are reserved in legacy functions.
Bram Moolenaar62aec932022-01-29 21:45:34 +00001901def s:WithReservedNames(firstline: string, lastline: string): string
Bram Moolenaarb816dae2020-09-20 22:04:00 +02001902 return firstline .. lastline
1903enddef
1904
1905def Test_argument_names()
1906 assert_equal('OK', WithReservedNames('O', 'K'))
1907enddef
1908
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001909def Test_call_func_defined_later()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001910 g:DefinedLater('one')->assert_equal('one')
Bram Moolenaar9bd5d872020-09-06 21:47:48 +02001911 assert_fails('NotDefined("one")', 'E117:', '', 2, 'Test_call_func_defined_later')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001912enddef
1913
Bram Moolenaar1df8b3f2020-04-23 18:13:23 +02001914func DefinedLater(arg)
1915 return a:arg
1916endfunc
1917
1918def Test_call_funcref()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001919 g:SomeFunc('abc')->assert_equal(3)
Bram Moolenaar9bd5d872020-09-06 21:47:48 +02001920 assert_fails('NotAFunc()', 'E117:', '', 2, 'Test_call_funcref') # comment after call
Bram Moolenaar2ef91562021-12-11 16:14:07 +00001921 assert_fails('g:NotAFunc()', 'E1085:', '', 3, 'Test_call_funcref')
Bram Moolenaar2f1980f2020-07-22 19:30:06 +02001922
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001923 var lines =<< trim END
Bram Moolenaar2f1980f2020-07-22 19:30:06 +02001924 vim9script
1925 def RetNumber(): number
1926 return 123
1927 enddef
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001928 var Funcref: func: number = function('RetNumber')
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001929 Funcref()->assert_equal(123)
Bram Moolenaar2f1980f2020-07-22 19:30:06 +02001930 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001931 v9.CheckScriptSuccess(lines)
Bram Moolenaar0f60e802020-07-22 20:16:11 +02001932
1933 lines =<< trim END
1934 vim9script
1935 def RetNumber(): number
1936 return 123
1937 enddef
1938 def Bar(F: func: number): number
1939 return F()
1940 enddef
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001941 var Funcref = function('RetNumber')
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001942 Bar(Funcref)->assert_equal(123)
Bram Moolenaar0f60e802020-07-22 20:16:11 +02001943 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001944 v9.CheckScriptSuccess(lines)
Bram Moolenaarbfba8652020-07-23 20:09:10 +02001945
1946 lines =<< trim END
1947 vim9script
1948 def UseNumber(nr: number)
1949 echo nr
1950 enddef
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001951 var Funcref: func(number) = function('UseNumber')
Bram Moolenaarbfba8652020-07-23 20:09:10 +02001952 Funcref(123)
1953 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001954 v9.CheckScriptSuccess(lines)
Bram Moolenaarb8070e32020-07-23 20:56:04 +02001955
1956 lines =<< trim END
1957 vim9script
1958 def UseNumber(nr: number)
1959 echo nr
1960 enddef
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001961 var Funcref: func(string) = function('UseNumber')
Bram Moolenaarb8070e32020-07-23 20:56:04 +02001962 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001963 v9.CheckScriptFailure(lines, 'E1012: Type mismatch; expected func(string) but got func(number)')
Bram Moolenaar4fc224c2020-07-26 17:56:25 +02001964
1965 lines =<< trim END
1966 vim9script
1967 def EchoNr(nr = 34)
1968 g:echo = nr
1969 enddef
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001970 var Funcref: func(?number) = function('EchoNr')
Bram Moolenaar4fc224c2020-07-26 17:56:25 +02001971 Funcref()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001972 g:echo->assert_equal(34)
Bram Moolenaar4fc224c2020-07-26 17:56:25 +02001973 Funcref(123)
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001974 g:echo->assert_equal(123)
Bram Moolenaar4fc224c2020-07-26 17:56:25 +02001975 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001976 v9.CheckScriptSuccess(lines)
Bram Moolenaarace61322020-07-26 18:16:58 +02001977
1978 lines =<< trim END
1979 vim9script
1980 def EchoList(...l: list<number>)
1981 g:echo = l
1982 enddef
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001983 var Funcref: func(...list<number>) = function('EchoList')
Bram Moolenaarace61322020-07-26 18:16:58 +02001984 Funcref()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001985 g:echo->assert_equal([])
Bram Moolenaarace61322020-07-26 18:16:58 +02001986 Funcref(1, 2, 3)
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001987 g:echo->assert_equal([1, 2, 3])
Bram Moolenaarace61322020-07-26 18:16:58 +02001988 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001989 v9.CheckScriptSuccess(lines)
Bram Moolenaar01865ad2020-07-26 18:33:09 +02001990
1991 lines =<< trim END
1992 vim9script
1993 def OptAndVar(nr: number, opt = 12, ...l: list<number>): number
1994 g:optarg = opt
1995 g:listarg = l
1996 return nr
1997 enddef
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001998 var Funcref: func(number, ?number, ...list<number>): number = function('OptAndVar')
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001999 Funcref(10)->assert_equal(10)
2000 g:optarg->assert_equal(12)
2001 g:listarg->assert_equal([])
Bram Moolenaar01865ad2020-07-26 18:33:09 +02002002
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002003 Funcref(11, 22)->assert_equal(11)
2004 g:optarg->assert_equal(22)
2005 g:listarg->assert_equal([])
Bram Moolenaar01865ad2020-07-26 18:33:09 +02002006
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002007 Funcref(17, 18, 1, 2, 3)->assert_equal(17)
2008 g:optarg->assert_equal(18)
2009 g:listarg->assert_equal([1, 2, 3])
Bram Moolenaar01865ad2020-07-26 18:33:09 +02002010 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002011 v9.CheckScriptSuccess(lines)
Kota Kato948a3892022-08-16 16:09:59 +01002012
2013 lines =<< trim END
2014 function s:func(num)
2015 return a:num * 2
2016 endfunction
2017
2018 def s:CallFuncref()
2019 var Funcref = function('s:func')
2020 Funcref(3)->assert_equal(6)
2021 enddef
2022 call s:CallFuncref()
2023 END
2024 v9.CheckScriptSuccess(lines)
2025
2026 lines =<< trim END
2027 function s:func(num)
2028 return a:num * 2
2029 endfunction
2030
2031 def s:CallFuncref()
2032 var Funcref = function(s:func)
2033 Funcref(3)->assert_equal(6)
2034 enddef
2035 call s:CallFuncref()
2036 END
2037 v9.CheckScriptSuccess(lines)
2038
2039 lines =<< trim END
2040 function s:func(num)
2041 return a:num * 2
2042 endfunction
2043
2044 def s:CallFuncref()
2045 var Funcref = s:func
2046 Funcref(3)->assert_equal(6)
2047 enddef
2048 call s:CallFuncref()
2049 END
2050 v9.CheckScriptSuccess(lines)
Bram Moolenaar1df8b3f2020-04-23 18:13:23 +02002051enddef
2052
2053let SomeFunc = function('len')
2054let NotAFunc = 'text'
2055
Bram Moolenaar99aaf0c2020-04-12 14:39:53 +02002056def CombineFuncrefTypes()
Bram Moolenaarf5be8cd2020-07-17 20:36:00 +02002057 # same arguments, different return type
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002058 var Ref1: func(bool): string
2059 var Ref2: func(bool): number
2060 var Ref3: func(bool): any
Bram Moolenaar99aaf0c2020-04-12 14:39:53 +02002061 Ref3 = g:cond ? Ref1 : Ref2
2062
Bram Moolenaarf5be8cd2020-07-17 20:36:00 +02002063 # different number of arguments
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002064 var Refa1: func(bool): number
2065 var Refa2: func(bool, number): number
2066 var Refa3: func: number
Bram Moolenaar99aaf0c2020-04-12 14:39:53 +02002067 Refa3 = g:cond ? Refa1 : Refa2
2068
Bram Moolenaarf5be8cd2020-07-17 20:36:00 +02002069 # different argument types
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002070 var Refb1: func(bool, string): number
2071 var Refb2: func(string, number): number
2072 var Refb3: func(any, any): number
Bram Moolenaar99aaf0c2020-04-12 14:39:53 +02002073 Refb3 = g:cond ? Refb1 : Refb2
2074enddef
2075
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002076def FuncWithForwardCall()
Bram Moolenaar1df8b3f2020-04-23 18:13:23 +02002077 return g:DefinedEvenLater("yes")
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002078enddef
2079
2080def DefinedEvenLater(arg: string): string
2081 return arg
2082enddef
2083
2084def Test_error_in_nested_function()
Bram Moolenaarf5be8cd2020-07-17 20:36:00 +02002085 # Error in called function requires unwinding the call stack.
Bram Moolenaar848fadd2022-01-30 15:28:30 +00002086 assert_fails('g:FuncWithForwardCall()', 'E1096:', '', 1, 'FuncWithForwardCall')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002087enddef
2088
Bram Moolenaar4bf10062021-12-28 17:23:12 +00002089def Test_nested_function_with_nextcmd()
Bram Moolenaar9c23f9b2021-12-26 14:23:22 +00002090 var lines =<< trim END
2091 vim9script
2092 # Define an outer function
2093 def FirstFunction()
2094 # Define an inner function
2095 def SecondFunction()
2096 # the function has a body, a double free is detected.
2097 AAAAA
2098
2099 # enddef followed by | or } followed by # one or more characters
2100 enddef|BBBB
2101 enddef
2102
2103 # Compile all functions
2104 defcompile
2105 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002106 v9.CheckScriptFailure(lines, 'E1173: Text found after enddef: BBBB')
Bram Moolenaar9c23f9b2021-12-26 14:23:22 +00002107enddef
2108
Bram Moolenaar4bf10062021-12-28 17:23:12 +00002109def Test_nested_function_with_args_split()
2110 var lines =<< trim END
2111 vim9script
2112 def FirstFunction()
2113 def SecondFunction(
2114 )
2115 # had a double free if the right parenthesis of the nested function is
2116 # on the next line
2117
2118 enddef|BBBB
2119 enddef
2120 # Compile all functions
2121 defcompile
2122 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002123 v9.CheckScriptFailure(lines, 'E1173: Text found after enddef: BBBB')
Bram Moolenaar7473a842021-12-28 17:55:26 +00002124
2125 lines =<< trim END
2126 vim9script
2127 def FirstFunction()
2128 func SecondFunction()
2129 endfunc|BBBB
2130 enddef
2131 defcompile
2132 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002133 v9.CheckScriptFailure(lines, 'E1173: Text found after endfunction: BBBB')
Bram Moolenaar4bf10062021-12-28 17:23:12 +00002134enddef
2135
Bram Moolenaar9f1a39a2022-01-08 15:39:39 +00002136def Test_error_in_function_args()
2137 var lines =<< trim END
2138 def FirstFunction()
2139 def SecondFunction(J =
2140 # Nois
2141 # one
2142
2143 enddef|BBBB
2144 enddef
2145 # Compile all functions
2146 defcompile
2147 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002148 v9.CheckScriptFailure(lines, 'E488:')
Bram Moolenaar9f1a39a2022-01-08 15:39:39 +00002149enddef
2150
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002151def Test_return_type_wrong()
Bram Moolenaar62aec932022-01-29 21:45:34 +00002152 v9.CheckScriptFailure([
Bram Moolenaar5a849da2020-08-08 16:47:30 +02002153 'def Func(): number',
2154 'return "a"',
2155 'enddef',
2156 'defcompile'], 'expected number but got string')
Bram Moolenaar2d870f82020-12-05 13:41:01 +01002157 delfunc! g:Func
Bram Moolenaar62aec932022-01-29 21:45:34 +00002158 v9.CheckScriptFailure([
Bram Moolenaar5a849da2020-08-08 16:47:30 +02002159 'def Func(): string',
2160 'return 1',
2161 'enddef',
2162 'defcompile'], 'expected string but got number')
Bram Moolenaar2d870f82020-12-05 13:41:01 +01002163 delfunc! g:Func
Bram Moolenaar62aec932022-01-29 21:45:34 +00002164 v9.CheckScriptFailure([
Bram Moolenaar5a849da2020-08-08 16:47:30 +02002165 'def Func(): void',
2166 'return "a"',
2167 'enddef',
2168 'defcompile'],
2169 'E1096: Returning a value in a function without a return type')
Bram Moolenaar2d870f82020-12-05 13:41:01 +01002170 delfunc! g:Func
Bram Moolenaar62aec932022-01-29 21:45:34 +00002171 v9.CheckScriptFailure([
Bram Moolenaar5a849da2020-08-08 16:47:30 +02002172 'def Func()',
2173 'return "a"',
2174 'enddef',
2175 'defcompile'],
2176 'E1096: Returning a value in a function without a return type')
Bram Moolenaar2d870f82020-12-05 13:41:01 +01002177 delfunc! g:Func
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002178
Bram Moolenaar62aec932022-01-29 21:45:34 +00002179 v9.CheckScriptFailure([
Bram Moolenaar5a849da2020-08-08 16:47:30 +02002180 'def Func(): number',
2181 'return',
2182 'enddef',
2183 'defcompile'], 'E1003:')
Bram Moolenaar2d870f82020-12-05 13:41:01 +01002184 delfunc! g:Func
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002185
Bram Moolenaar62aec932022-01-29 21:45:34 +00002186 v9.CheckScriptFailure([
Bram Moolenaar33ea9fd2021-08-08 19:07:37 +02002187 'def Func():number',
2188 'return 123',
2189 'enddef',
2190 'defcompile'], 'E1069:')
2191 delfunc! g:Func
2192
Bram Moolenaar62aec932022-01-29 21:45:34 +00002193 v9.CheckScriptFailure([
Bram Moolenaar33ea9fd2021-08-08 19:07:37 +02002194 'def Func() :number',
2195 'return 123',
2196 'enddef',
2197 'defcompile'], 'E1059:')
2198 delfunc! g:Func
2199
Bram Moolenaar62aec932022-01-29 21:45:34 +00002200 v9.CheckScriptFailure([
Bram Moolenaar33ea9fd2021-08-08 19:07:37 +02002201 'def Func() : number',
2202 'return 123',
2203 'enddef',
2204 'defcompile'], 'E1059:')
2205 delfunc! g:Func
2206
Bram Moolenaar62e0e2e2022-08-20 12:07:58 +01002207 v9.CheckScriptFailure(['def Func(): list', 'return []', 'enddef'], 'E1008: Missing <type> after list')
Bram Moolenaar2d870f82020-12-05 13:41:01 +01002208 delfunc! g:Func
Bram Moolenaar62e0e2e2022-08-20 12:07:58 +01002209 v9.CheckScriptFailure(['def Func(): dict', 'return {}', 'enddef'], 'E1008: Missing <type> after dict')
Bram Moolenaar2d870f82020-12-05 13:41:01 +01002210 delfunc! g:Func
Bram Moolenaar62aec932022-01-29 21:45:34 +00002211 v9.CheckScriptFailure(['def Func()', 'return 1'], 'E1057:')
Bram Moolenaar2d870f82020-12-05 13:41:01 +01002212 delfunc! g:Func
Bram Moolenaar5a849da2020-08-08 16:47:30 +02002213
Bram Moolenaar62aec932022-01-29 21:45:34 +00002214 v9.CheckScriptFailure([
Bram Moolenaar5a849da2020-08-08 16:47:30 +02002215 'vim9script',
2216 'def FuncB()',
2217 ' return 123',
2218 'enddef',
2219 'def FuncA()',
2220 ' FuncB()',
2221 'enddef',
2222 'defcompile'], 'E1096:')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002223enddef
2224
2225def Test_arg_type_wrong()
Bram Moolenaar62e0e2e2022-08-20 12:07:58 +01002226 v9.CheckScriptFailure(['def Func3(items: list)', 'echo "a"', 'enddef'], 'E1008: Missing <type> after list')
Bram Moolenaar62aec932022-01-29 21:45:34 +00002227 v9.CheckScriptFailure(['def Func4(...)', 'echo "a"', 'enddef'], 'E1055: Missing name after ...')
2228 v9.CheckScriptFailure(['def Func5(items:string)', 'echo "a"'], 'E1069:')
2229 v9.CheckScriptFailure(['def Func5(items)', 'echo "a"'], 'E1077:')
2230 v9.CheckScriptFailure(['def Func6(...x:list<number>)', 'echo "a"', 'enddef'], 'E1069:')
2231 v9.CheckScriptFailure(['def Func7(...x: int)', 'echo "a"', 'enddef'], 'E1010:')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002232enddef
2233
Bram Moolenaar86cdb8a2021-04-06 19:01:03 +02002234def Test_white_space_before_comma()
2235 var lines =<< trim END
2236 vim9script
2237 def Func(a: number , b: number)
2238 enddef
2239 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002240 v9.CheckScriptFailure(lines, 'E1068:')
Yegappan Lakshmanan611728f2021-05-24 15:15:47 +02002241 call assert_fails('vim9cmd echo stridx("a" .. "b" , "a")', 'E1068:')
Bram Moolenaar86cdb8a2021-04-06 19:01:03 +02002242enddef
2243
Bram Moolenaar608d78f2021-03-06 22:33:12 +01002244def Test_white_space_after_comma()
2245 var lines =<< trim END
2246 vim9script
2247 def Func(a: number,b: number)
2248 enddef
2249 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002250 v9.CheckScriptFailure(lines, 'E1069:')
Bram Moolenaar608d78f2021-03-06 22:33:12 +01002251
2252 # OK in legacy function
2253 lines =<< trim END
2254 vim9script
2255 func Func(a,b)
2256 endfunc
2257 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002258 v9.CheckScriptSuccess(lines)
Bram Moolenaar608d78f2021-03-06 22:33:12 +01002259enddef
2260
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002261def Test_vim9script_call()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002262 var lines =<< trim END
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002263 vim9script
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002264 var name = ''
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002265 def MyFunc(arg: string)
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002266 name = arg
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002267 enddef
2268 MyFunc('foobar')
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002269 name->assert_equal('foobar')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002270
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002271 var str = 'barfoo'
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002272 str->MyFunc()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002273 name->assert_equal('barfoo')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002274
Bram Moolenaar67979662020-06-20 22:50:47 +02002275 g:value = 'value'
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002276 g:value->MyFunc()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002277 name->assert_equal('value')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002278
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002279 var listvar = []
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002280 def ListFunc(arg: list<number>)
2281 listvar = arg
2282 enddef
2283 [1, 2, 3]->ListFunc()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002284 listvar->assert_equal([1, 2, 3])
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002285
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002286 var dictvar = {}
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002287 def DictFunc(arg: dict<number>)
2288 dictvar = arg
2289 enddef
Bram Moolenaare0de1712020-12-02 17:36:54 +01002290 {a: 1, b: 2}->DictFunc()
2291 dictvar->assert_equal({a: 1, b: 2})
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002292 def CompiledDict()
Bram Moolenaare0de1712020-12-02 17:36:54 +01002293 {a: 3, b: 4}->DictFunc()
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002294 enddef
2295 CompiledDict()
Bram Moolenaare0de1712020-12-02 17:36:54 +01002296 dictvar->assert_equal({a: 3, b: 4})
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002297
Bram Moolenaare0de1712020-12-02 17:36:54 +01002298 {a: 3, b: 4}->DictFunc()
2299 dictvar->assert_equal({a: 3, b: 4})
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002300
2301 ('text')->MyFunc()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002302 name->assert_equal('text')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002303 ("some")->MyFunc()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002304 name->assert_equal('some')
Bram Moolenaare6b53242020-07-01 17:28:33 +02002305
Bram Moolenaar13e12b82020-07-24 18:47:22 +02002306 # line starting with single quote is not a mark
Bram Moolenaar10409562020-07-29 20:00:38 +02002307 # line starting with double quote can be a method call
Bram Moolenaar3d48e252020-07-15 14:15:52 +02002308 'asdfasdf'->MyFunc()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002309 name->assert_equal('asdfasdf')
Bram Moolenaar10409562020-07-29 20:00:38 +02002310 "xyz"->MyFunc()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002311 name->assert_equal('xyz')
Bram Moolenaar3d48e252020-07-15 14:15:52 +02002312
2313 def UseString()
2314 'xyork'->MyFunc()
2315 enddef
2316 UseString()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002317 name->assert_equal('xyork')
Bram Moolenaar3d48e252020-07-15 14:15:52 +02002318
Bram Moolenaar10409562020-07-29 20:00:38 +02002319 def UseString2()
2320 "knife"->MyFunc()
2321 enddef
2322 UseString2()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002323 name->assert_equal('knife')
Bram Moolenaar10409562020-07-29 20:00:38 +02002324
Bram Moolenaar13e12b82020-07-24 18:47:22 +02002325 # prepending a colon makes it a mark
2326 new
2327 setline(1, ['aaa', 'bbb', 'ccc'])
2328 normal! 3Gmt1G
2329 :'t
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002330 getcurpos()[1]->assert_equal(3)
Bram Moolenaar13e12b82020-07-24 18:47:22 +02002331 bwipe!
2332
Bram Moolenaare6b53242020-07-01 17:28:33 +02002333 MyFunc(
2334 'continued'
2335 )
2336 assert_equal('continued',
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002337 name
Bram Moolenaare6b53242020-07-01 17:28:33 +02002338 )
2339
2340 call MyFunc(
2341 'more'
2342 ..
2343 'lines'
2344 )
2345 assert_equal(
2346 'morelines',
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002347 name)
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002348 END
Bram Moolenaarf5fec052022-09-11 11:49:22 +01002349 writefile(lines, 'Xcall.vim', 'D')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002350 source Xcall.vim
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002351enddef
2352
2353def Test_vim9script_call_fail_decl()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002354 var lines =<< trim END
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002355 vim9script
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002356 var name = ''
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002357 def MyFunc(arg: string)
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002358 var name = 123
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002359 enddef
Bram Moolenaar822ba242020-05-24 23:00:18 +02002360 defcompile
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002361 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002362 v9.CheckScriptFailure(lines, 'E1054:')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002363enddef
2364
Bram Moolenaar65b95452020-07-19 14:03:09 +02002365def Test_vim9script_call_fail_type()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002366 var lines =<< trim END
Bram Moolenaar65b95452020-07-19 14:03:09 +02002367 vim9script
2368 def MyFunc(arg: string)
2369 echo arg
2370 enddef
2371 MyFunc(1234)
2372 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002373 v9.CheckScriptFailure(lines, 'E1013: Argument 1: type mismatch, expected string but got number')
Bram Moolenaar65b95452020-07-19 14:03:09 +02002374enddef
2375
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002376def Test_vim9script_call_fail_const()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002377 var lines =<< trim END
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002378 vim9script
2379 const var = ''
2380 def MyFunc(arg: string)
2381 var = 'asdf'
2382 enddef
Bram Moolenaar822ba242020-05-24 23:00:18 +02002383 defcompile
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002384 END
Bram Moolenaarf5fec052022-09-11 11:49:22 +01002385 writefile(lines, 'Xcall_const.vim', 'D')
Bram Moolenaar9bd5d872020-09-06 21:47:48 +02002386 assert_fails('source Xcall_const.vim', 'E46:', '', 1, 'MyFunc')
Bram Moolenaar3bdc90b2020-12-22 20:35:40 +01002387
2388 lines =<< trim END
2389 const g:Aconst = 77
2390 def Change()
2391 # comment
2392 g:Aconst = 99
2393 enddef
2394 call Change()
2395 unlet g:Aconst
2396 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002397 v9.CheckScriptFailure(lines, 'E741: Value is locked: Aconst', 2)
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002398enddef
2399
2400" Test that inside :function a Python function can be defined, :def is not
2401" recognized.
2402func Test_function_python()
2403 CheckFeature python3
Bram Moolenaar727345e2020-09-27 23:33:59 +02002404 let py = 'python3'
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002405 execute py "<< EOF"
2406def do_something():
2407 return 1
2408EOF
2409endfunc
2410
2411def Test_delfunc()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002412 var lines =<< trim END
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002413 vim9script
Bram Moolenaar4c17ad92020-04-27 22:47:51 +02002414 def g:GoneSoon()
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002415 echo 'hello'
2416 enddef
2417
2418 def CallGoneSoon()
Bram Moolenaar62aec932022-01-29 21:45:34 +00002419 g:GoneSoon()
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002420 enddef
Bram Moolenaar822ba242020-05-24 23:00:18 +02002421 defcompile
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002422
Bram Moolenaar4c17ad92020-04-27 22:47:51 +02002423 delfunc g:GoneSoon
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002424 CallGoneSoon()
2425 END
Bram Moolenaarf5fec052022-09-11 11:49:22 +01002426 writefile(lines, 'XToDelFunc', 'D')
Bram Moolenaar9bd5d872020-09-06 21:47:48 +02002427 assert_fails('so XToDelFunc', 'E933:', '', 1, 'CallGoneSoon')
2428 assert_fails('so XToDelFunc', 'E933:', '', 1, 'CallGoneSoon')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002429enddef
2430
Bram Moolenaar7509ad82021-12-14 18:14:37 +00002431func Test_free_dict_while_in_funcstack()
2432 " relies on the sleep command
2433 CheckUnix
2434 call Run_Test_free_dict_while_in_funcstack()
2435endfunc
2436
2437def Run_Test_free_dict_while_in_funcstack()
Bram Moolenaar7509ad82021-12-14 18:14:37 +00002438 # this was freeing the TermRun() default argument dictionary while it was
2439 # still referenced in a funcstack_T
2440 var lines =<< trim END
2441 vim9script
2442
2443 &updatetime = 400
2444 def TermRun(_ = {})
2445 def Post()
2446 enddef
2447 def Exec()
2448 term_start('sleep 1', {
2449 term_finish: 'close',
2450 exit_cb: (_, _) => Post(),
2451 })
2452 enddef
2453 Exec()
2454 enddef
2455 nnoremap <F4> <Cmd>call <SID>TermRun()<CR>
2456 timer_start(100, (_) => feedkeys("\<F4>"))
2457 timer_start(1000, (_) => feedkeys("\<F4>"))
2458 sleep 1500m
2459 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002460 v9.CheckScriptSuccess(lines)
Bram Moolenaar7509ad82021-12-14 18:14:37 +00002461 nunmap <F4>
2462 set updatetime&
2463enddef
2464
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002465def Test_redef_failure()
Bram Moolenaard2c61702020-09-06 15:58:36 +02002466 writefile(['def Func0(): string', 'return "Func0"', 'enddef'], 'Xdef')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002467 so Xdef
Bram Moolenaard2c61702020-09-06 15:58:36 +02002468 writefile(['def Func1(): string', 'return "Func1"', 'enddef'], 'Xdef')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002469 so Xdef
Bram Moolenaard2c61702020-09-06 15:58:36 +02002470 writefile(['def! Func0(): string', 'enddef', 'defcompile'], 'Xdef')
Bram Moolenaar9bd5d872020-09-06 21:47:48 +02002471 assert_fails('so Xdef', 'E1027:', '', 1, 'Func0')
Bram Moolenaard2c61702020-09-06 15:58:36 +02002472 writefile(['def Func2(): string', 'return "Func2"', 'enddef'], 'Xdef')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002473 so Xdef
Bram Moolenaard2c61702020-09-06 15:58:36 +02002474 delete('Xdef')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002475
Bram Moolenaar701cc6c2021-04-10 13:33:48 +02002476 assert_fails('g:Func0()', 'E1091:')
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002477 g:Func1()->assert_equal('Func1')
2478 g:Func2()->assert_equal('Func2')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002479
2480 delfunc! Func0
2481 delfunc! Func1
2482 delfunc! Func2
2483enddef
2484
Bram Moolenaarf93c7fe2020-04-23 22:16:53 +02002485def Test_vim9script_func()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002486 var lines =<< trim END
Bram Moolenaarf93c7fe2020-04-23 22:16:53 +02002487 vim9script
2488 func Func(arg)
2489 echo a:arg
2490 endfunc
2491 Func('text')
2492 END
Bram Moolenaarf5fec052022-09-11 11:49:22 +01002493 writefile(lines, 'XVim9Func', 'D')
Bram Moolenaarf93c7fe2020-04-23 22:16:53 +02002494 so XVim9Func
Bram Moolenaarf93c7fe2020-04-23 22:16:53 +02002495enddef
2496
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002497let s:funcResult = 0
2498
2499def FuncNoArgNoRet()
Bram Moolenaar53900992020-08-22 19:02:02 +02002500 s:funcResult = 11
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002501enddef
2502
2503def FuncNoArgRetNumber(): number
Bram Moolenaar53900992020-08-22 19:02:02 +02002504 s:funcResult = 22
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002505 return 1234
2506enddef
2507
Bram Moolenaarec5929d2020-04-07 20:53:39 +02002508def FuncNoArgRetString(): string
Bram Moolenaar53900992020-08-22 19:02:02 +02002509 s:funcResult = 45
Bram Moolenaarec5929d2020-04-07 20:53:39 +02002510 return 'text'
2511enddef
2512
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002513def FuncOneArgNoRet(arg: number)
Bram Moolenaar53900992020-08-22 19:02:02 +02002514 s:funcResult = arg
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002515enddef
2516
2517def FuncOneArgRetNumber(arg: number): number
Bram Moolenaar53900992020-08-22 19:02:02 +02002518 s:funcResult = arg
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002519 return arg
2520enddef
2521
Bram Moolenaar08938ee2020-04-11 23:17:17 +02002522def FuncTwoArgNoRet(one: bool, two: number)
Bram Moolenaar53900992020-08-22 19:02:02 +02002523 s:funcResult = two
Bram Moolenaar08938ee2020-04-11 23:17:17 +02002524enddef
2525
Bram Moolenaar62aec932022-01-29 21:45:34 +00002526def s:FuncOneArgRetString(arg: string): string
Bram Moolenaarec5929d2020-04-07 20:53:39 +02002527 return arg
2528enddef
2529
Bram Moolenaar62aec932022-01-29 21:45:34 +00002530def s:FuncOneArgRetAny(arg: any): any
Bram Moolenaar89228602020-04-05 22:14:54 +02002531 return arg
2532enddef
2533
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002534def Test_func_type()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002535 var Ref1: func()
Bram Moolenaar53900992020-08-22 19:02:02 +02002536 s:funcResult = 0
Bram Moolenaar62aec932022-01-29 21:45:34 +00002537 Ref1 = g:FuncNoArgNoRet
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002538 Ref1()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002539 s:funcResult->assert_equal(11)
Bram Moolenaar4c683752020-04-05 21:38:23 +02002540
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002541 var Ref2: func
Bram Moolenaar53900992020-08-22 19:02:02 +02002542 s:funcResult = 0
Bram Moolenaar62aec932022-01-29 21:45:34 +00002543 Ref2 = g:FuncNoArgNoRet
Bram Moolenaar4c683752020-04-05 21:38:23 +02002544 Ref2()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002545 s:funcResult->assert_equal(11)
Bram Moolenaar4c683752020-04-05 21:38:23 +02002546
Bram Moolenaar53900992020-08-22 19:02:02 +02002547 s:funcResult = 0
Bram Moolenaar62aec932022-01-29 21:45:34 +00002548 Ref2 = g:FuncOneArgNoRet
Bram Moolenaar4c683752020-04-05 21:38:23 +02002549 Ref2(12)
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002550 s:funcResult->assert_equal(12)
Bram Moolenaar4c683752020-04-05 21:38:23 +02002551
Bram Moolenaar53900992020-08-22 19:02:02 +02002552 s:funcResult = 0
Bram Moolenaar62aec932022-01-29 21:45:34 +00002553 Ref2 = g:FuncNoArgRetNumber
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002554 Ref2()->assert_equal(1234)
2555 s:funcResult->assert_equal(22)
Bram Moolenaar4c683752020-04-05 21:38:23 +02002556
Bram Moolenaar53900992020-08-22 19:02:02 +02002557 s:funcResult = 0
Bram Moolenaar848fadd2022-01-30 15:28:30 +00002558 Ref2 = g:FuncOneArgRetNumber
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002559 Ref2(13)->assert_equal(13)
2560 s:funcResult->assert_equal(13)
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002561enddef
2562
Bram Moolenaar9978d472020-07-05 16:01:56 +02002563def Test_repeat_return_type()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002564 var res = 0
Bram Moolenaar9978d472020-07-05 16:01:56 +02002565 for n in repeat([1], 3)
2566 res += n
2567 endfor
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002568 res->assert_equal(3)
Bram Moolenaarfce82b32020-07-05 16:07:21 +02002569
2570 res = 0
Bakudankun375141e2022-09-09 18:46:47 +01002571 for n in repeat(0z01, 3)->blob2list()
2572 res += n
2573 endfor
2574 res->assert_equal(3)
2575
2576 res = 0
Bram Moolenaarfce82b32020-07-05 16:07:21 +02002577 for n in add([1, 2], 3)
2578 res += n
2579 endfor
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002580 res->assert_equal(6)
Bram Moolenaar9978d472020-07-05 16:01:56 +02002581enddef
2582
Bram Moolenaar846178a2020-07-05 17:04:13 +02002583def Test_argv_return_type()
2584 next fileone filetwo
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002585 var res = ''
Bram Moolenaar846178a2020-07-05 17:04:13 +02002586 for name in argv()
2587 res ..= name
2588 endfor
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002589 res->assert_equal('fileonefiletwo')
Bram Moolenaar846178a2020-07-05 17:04:13 +02002590enddef
2591
Bram Moolenaarec5929d2020-04-07 20:53:39 +02002592def Test_func_type_part()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002593 var RefVoid: func: void
Bram Moolenaar62aec932022-01-29 21:45:34 +00002594 RefVoid = g:FuncNoArgNoRet
2595 RefVoid = g:FuncOneArgNoRet
2596 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 +00002597 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 +02002598
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002599 var RefAny: func(): any
Bram Moolenaar62aec932022-01-29 21:45:34 +00002600 RefAny = g:FuncNoArgRetNumber
Bram Moolenaar848fadd2022-01-30 15:28:30 +00002601 RefAny = g:FuncNoArgRetString
Bram Moolenaar62aec932022-01-29 21:45:34 +00002602 v9.CheckDefFailure(['var RefAny: func(): any', 'RefAny = g:FuncNoArgNoRet'], 'E1012: Type mismatch; expected func(): any but got func()')
2603 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 +02002604
Bram Moolenaar6abd3dc2020-10-04 14:17:32 +02002605 var RefAnyNoArgs: func: any = RefAny
2606
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002607 var RefNr: func: number
Bram Moolenaar62aec932022-01-29 21:45:34 +00002608 RefNr = g:FuncNoArgRetNumber
Bram Moolenaar848fadd2022-01-30 15:28:30 +00002609 RefNr = g:FuncOneArgRetNumber
Bram Moolenaar62aec932022-01-29 21:45:34 +00002610 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 +00002611 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 +02002612
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002613 var RefStr: func: string
Bram Moolenaar848fadd2022-01-30 15:28:30 +00002614 RefStr = g:FuncNoArgRetString
Bram Moolenaarec5929d2020-04-07 20:53:39 +02002615 RefStr = FuncOneArgRetString
Bram Moolenaar62aec932022-01-29 21:45:34 +00002616 v9.CheckDefFailure(['var RefStr: func: string', 'RefStr = g:FuncNoArgNoRet'], 'E1012: Type mismatch; expected func(...): string but got func()')
2617 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 +02002618enddef
2619
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002620def Test_func_type_fails()
Bram Moolenaar62aec932022-01-29 21:45:34 +00002621 v9.CheckDefFailure(['var ref1: func()'], 'E704:')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002622
Bram Moolenaar62aec932022-01-29 21:45:34 +00002623 v9.CheckDefFailure(['var Ref1: func()', 'Ref1 = g:FuncNoArgRetNumber'], 'E1012: Type mismatch; expected func() but got func(): number')
2624 v9.CheckDefFailure(['var Ref1: func()', 'Ref1 = g:FuncOneArgNoRet'], 'E1012: Type mismatch; expected func() but got func(number)')
Bram Moolenaar848fadd2022-01-30 15:28:30 +00002625 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 +00002626 v9.CheckDefFailure(['var Ref1: func(bool)', 'Ref1 = g:FuncTwoArgNoRet'], 'E1012: Type mismatch; expected func(bool) but got func(bool, number)')
2627 v9.CheckDefFailure(['var Ref1: func(?bool)', 'Ref1 = g:FuncTwoArgNoRet'], 'E1012: Type mismatch; expected func(?bool) but got func(bool, number)')
2628 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 +02002629
Bram Moolenaar62aec932022-01-29 21:45:34 +00002630 v9.CheckDefFailure(['var RefWrong: func(string ,number)'], 'E1068:')
2631 v9.CheckDefFailure(['var RefWrong: func(string,number)'], 'E1069:')
2632 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:')
2633 v9.CheckDefFailure(['var RefWrong: func(bool):string'], 'E1069:')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002634enddef
2635
Bram Moolenaar89228602020-04-05 22:14:54 +02002636def Test_func_return_type()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002637 var nr: number
Bram Moolenaar62aec932022-01-29 21:45:34 +00002638 nr = g:FuncNoArgRetNumber()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002639 nr->assert_equal(1234)
Bram Moolenaar89228602020-04-05 22:14:54 +02002640
2641 nr = FuncOneArgRetAny(122)
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002642 nr->assert_equal(122)
Bram Moolenaar89228602020-04-05 22:14:54 +02002643
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002644 var str: string
Bram Moolenaar89228602020-04-05 22:14:54 +02002645 str = FuncOneArgRetAny('yes')
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002646 str->assert_equal('yes')
Bram Moolenaar89228602020-04-05 22:14:54 +02002647
Bram Moolenaar62aec932022-01-29 21:45:34 +00002648 v9.CheckDefFailure(['var str: string', 'str = g:FuncNoArgRetNumber()'], 'E1012: Type mismatch; expected string but got number')
Bram Moolenaar89228602020-04-05 22:14:54 +02002649enddef
2650
Bram Moolenaar6abd3dc2020-10-04 14:17:32 +02002651def Test_func_common_type()
2652 def FuncOne(n: number): number
2653 return n
2654 enddef
2655 def FuncTwo(s: string): number
2656 return len(s)
2657 enddef
2658 def FuncThree(n: number, s: string): number
2659 return n + len(s)
2660 enddef
2661 var list = [FuncOne, FuncTwo, FuncThree]
2662 assert_equal(8, list[0](8))
2663 assert_equal(4, list[1]('word'))
2664 assert_equal(7, list[2](3, 'word'))
2665enddef
2666
Bram Moolenaar62aec932022-01-29 21:45:34 +00002667def s:MultiLine(
Bram Moolenaar5e774c72020-04-12 21:53:00 +02002668 arg1: string,
2669 arg2 = 1234,
2670 ...rest: list<string>
2671 ): string
2672 return arg1 .. arg2 .. join(rest, '-')
2673enddef
2674
Bram Moolenaar2c330432020-04-13 14:41:35 +02002675def MultiLineComment(
2676 arg1: string, # comment
2677 arg2 = 1234, # comment
2678 ...rest: list<string> # comment
2679 ): string # comment
2680 return arg1 .. arg2 .. join(rest, '-')
2681enddef
2682
Bram Moolenaar5e774c72020-04-12 21:53:00 +02002683def Test_multiline()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002684 MultiLine('text')->assert_equal('text1234')
2685 MultiLine('text', 777)->assert_equal('text777')
2686 MultiLine('text', 777, 'one')->assert_equal('text777one')
2687 MultiLine('text', 777, 'one', 'two')->assert_equal('text777one-two')
Bram Moolenaar5e774c72020-04-12 21:53:00 +02002688enddef
2689
Bram Moolenaar23e03252020-04-12 22:22:31 +02002690func Test_multiline_not_vim9()
Bram Moolenaar62aec932022-01-29 21:45:34 +00002691 call s:MultiLine('text')->assert_equal('text1234')
2692 call s:MultiLine('text', 777)->assert_equal('text777')
2693 call s:MultiLine('text', 777, 'one')->assert_equal('text777one')
2694 call s:MultiLine('text', 777, 'one', 'two')->assert_equal('text777one-two')
Bram Moolenaar23e03252020-04-12 22:22:31 +02002695endfunc
2696
Bram Moolenaar5e774c72020-04-12 21:53:00 +02002697
Bram Moolenaaree4e0c12020-04-06 21:35:05 +02002698" When using CheckScriptFailure() for the below test, E1010 is generated instead
2699" of E1056.
2700func Test_E1056_1059()
2701 let caught_1056 = 0
2702 try
2703 def F():
2704 return 1
2705 enddef
2706 catch /E1056:/
2707 let caught_1056 = 1
2708 endtry
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002709 eval caught_1056->assert_equal(1)
Bram Moolenaaree4e0c12020-04-06 21:35:05 +02002710
2711 let caught_1059 = 0
2712 try
2713 def F5(items : list)
2714 echo 'a'
2715 enddef
2716 catch /E1059:/
2717 let caught_1059 = 1
2718 endtry
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002719 eval caught_1059->assert_equal(1)
Bram Moolenaaree4e0c12020-04-06 21:35:05 +02002720endfunc
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002721
Bram Moolenaar015f4262020-05-05 21:25:22 +02002722func DelMe()
2723 echo 'DelMe'
2724endfunc
2725
Bram Moolenaarbf8feb52020-08-08 14:26:31 +02002726def Test_error_reporting()
2727 # comment lines at the start of the function
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002728 var lines =<< trim END
Bram Moolenaarbf8feb52020-08-08 14:26:31 +02002729 " comment
2730 def Func()
2731 # comment
2732 # comment
2733 invalid
2734 enddef
2735 defcompile
2736 END
Bram Moolenaarf5fec052022-09-11 11:49:22 +01002737 writefile(lines, 'Xdef', 'D')
Bram Moolenaarbf8feb52020-08-08 14:26:31 +02002738 try
2739 source Xdef
Bram Moolenaar7517ffd2020-08-14 18:35:07 +02002740 assert_report('should have failed')
Bram Moolenaarbf8feb52020-08-08 14:26:31 +02002741 catch /E476:/
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002742 v:exception->assert_match('Invalid command: invalid')
2743 v:throwpoint->assert_match(', line 3$')
Bram Moolenaarbf8feb52020-08-08 14:26:31 +02002744 endtry
Bram Moolenaar2d870f82020-12-05 13:41:01 +01002745 delfunc! g:Func
Bram Moolenaarbf8feb52020-08-08 14:26:31 +02002746
2747 # comment lines after the start of the function
2748 lines =<< trim END
2749 " comment
2750 def Func()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002751 var x = 1234
Bram Moolenaarbf8feb52020-08-08 14:26:31 +02002752 # comment
2753 # comment
2754 invalid
2755 enddef
2756 defcompile
2757 END
Bram Moolenaar08052222020-09-14 17:04:31 +02002758 writefile(lines, 'Xdef')
Bram Moolenaarbf8feb52020-08-08 14:26:31 +02002759 try
2760 source Xdef
Bram Moolenaar7517ffd2020-08-14 18:35:07 +02002761 assert_report('should have failed')
Bram Moolenaarbf8feb52020-08-08 14:26:31 +02002762 catch /E476:/
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002763 v:exception->assert_match('Invalid command: invalid')
2764 v:throwpoint->assert_match(', line 4$')
Bram Moolenaarbf8feb52020-08-08 14:26:31 +02002765 endtry
Bram Moolenaar2d870f82020-12-05 13:41:01 +01002766 delfunc! g:Func
Bram Moolenaarbf8feb52020-08-08 14:26:31 +02002767
Bram Moolenaar7517ffd2020-08-14 18:35:07 +02002768 lines =<< trim END
2769 vim9script
2770 def Func()
Bram Moolenaare0de1712020-12-02 17:36:54 +01002771 var db = {foo: 1, bar: 2}
Bram Moolenaar7517ffd2020-08-14 18:35:07 +02002772 # comment
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002773 var x = db.asdf
Bram Moolenaar7517ffd2020-08-14 18:35:07 +02002774 enddef
2775 defcompile
2776 Func()
2777 END
Bram Moolenaar08052222020-09-14 17:04:31 +02002778 writefile(lines, 'Xdef')
Bram Moolenaar7517ffd2020-08-14 18:35:07 +02002779 try
2780 source Xdef
2781 assert_report('should have failed')
2782 catch /E716:/
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002783 v:throwpoint->assert_match('_Func, line 3$')
Bram Moolenaar7517ffd2020-08-14 18:35:07 +02002784 endtry
Bram Moolenaar2d870f82020-12-05 13:41:01 +01002785 delfunc! g:Func
Bram Moolenaarbf8feb52020-08-08 14:26:31 +02002786enddef
2787
Bram Moolenaar015f4262020-05-05 21:25:22 +02002788def Test_deleted_function()
Bram Moolenaar62aec932022-01-29 21:45:34 +00002789 v9.CheckDefExecFailure([
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002790 'var RefMe: func = function("g:DelMe")',
Bram Moolenaar015f4262020-05-05 21:25:22 +02002791 'delfunc g:DelMe',
2792 'echo RefMe()'], 'E117:')
2793enddef
2794
2795def Test_unknown_function()
Bram Moolenaar62aec932022-01-29 21:45:34 +00002796 v9.CheckDefExecFailure([
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002797 'var Ref: func = function("NotExist")',
Bram Moolenaar9b7bf9e2020-07-11 22:14:59 +02002798 'delfunc g:NotExist'], 'E700:')
Bram Moolenaar015f4262020-05-05 21:25:22 +02002799enddef
2800
Bram Moolenaar62aec932022-01-29 21:45:34 +00002801def s:RefFunc(Ref: func(any): any): string
Bram Moolenaarc8cd2b32020-05-01 19:29:08 +02002802 return Ref('more')
2803enddef
2804
2805def Test_closure_simple()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002806 var local = 'some '
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01002807 RefFunc((s) => local .. s)->assert_equal('some more')
Bram Moolenaarc8cd2b32020-05-01 19:29:08 +02002808enddef
2809
Bram Moolenaar62aec932022-01-29 21:45:34 +00002810def s:MakeRef()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002811 var local = 'some '
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01002812 g:Ref = (s) => local .. s
Bram Moolenaarbf67ea12020-05-02 17:52:42 +02002813enddef
2814
2815def Test_closure_ref_after_return()
2816 MakeRef()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002817 g:Ref('thing')->assert_equal('some thing')
Bram Moolenaarbf67ea12020-05-02 17:52:42 +02002818 unlet g:Ref
2819enddef
2820
Bram Moolenaar62aec932022-01-29 21:45:34 +00002821def s:MakeTwoRefs()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002822 var local = ['some']
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01002823 g:Extend = (s) => local->add(s)
2824 g:Read = () => local
Bram Moolenaar5adc55c2020-05-02 23:12:58 +02002825enddef
2826
2827def Test_closure_two_refs()
2828 MakeTwoRefs()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002829 join(g:Read(), ' ')->assert_equal('some')
Bram Moolenaar5adc55c2020-05-02 23:12:58 +02002830 g:Extend('more')
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002831 join(g:Read(), ' ')->assert_equal('some more')
Bram Moolenaar5adc55c2020-05-02 23:12:58 +02002832 g:Extend('even')
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002833 join(g:Read(), ' ')->assert_equal('some more even')
Bram Moolenaar5adc55c2020-05-02 23:12:58 +02002834
2835 unlet g:Extend
2836 unlet g:Read
2837enddef
2838
Bram Moolenaar62aec932022-01-29 21:45:34 +00002839def s:ReadRef(Ref: func(): list<string>): string
Bram Moolenaar5adc55c2020-05-02 23:12:58 +02002840 return join(Ref(), ' ')
2841enddef
2842
Bram Moolenaar62aec932022-01-29 21:45:34 +00002843def s:ExtendRef(Ref: func(string): list<string>, add: string)
Bram Moolenaar5adc55c2020-05-02 23:12:58 +02002844 Ref(add)
2845enddef
2846
2847def Test_closure_two_indirect_refs()
Bram Moolenaarf7779c62020-05-03 15:38:16 +02002848 MakeTwoRefs()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002849 ReadRef(g:Read)->assert_equal('some')
Bram Moolenaar5adc55c2020-05-02 23:12:58 +02002850 ExtendRef(g:Extend, 'more')
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002851 ReadRef(g:Read)->assert_equal('some more')
Bram Moolenaar5adc55c2020-05-02 23:12:58 +02002852 ExtendRef(g:Extend, 'even')
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002853 ReadRef(g:Read)->assert_equal('some more even')
Bram Moolenaar5adc55c2020-05-02 23:12:58 +02002854
2855 unlet g:Extend
2856 unlet g:Read
2857enddef
Bram Moolenaarbf67ea12020-05-02 17:52:42 +02002858
Bram Moolenaar62aec932022-01-29 21:45:34 +00002859def s:MakeArgRefs(theArg: string)
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002860 var local = 'loc_val'
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01002861 g:UseArg = (s) => theArg .. '/' .. local .. '/' .. s
Bram Moolenaar2fd4cd72020-05-03 22:30:49 +02002862enddef
2863
Bram Moolenaar62aec932022-01-29 21:45:34 +00002864def s:MakeArgRefsVarargs(theArg: string, ...rest: list<string>)
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002865 var local = 'the_loc'
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01002866 g:UseVararg = (s) => theArg .. '/' .. local .. '/' .. s .. '/' .. join(rest)
Bram Moolenaar2fd4cd72020-05-03 22:30:49 +02002867enddef
2868
2869def Test_closure_using_argument()
2870 MakeArgRefs('arg_val')
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002871 g:UseArg('call_val')->assert_equal('arg_val/loc_val/call_val')
Bram Moolenaar2fd4cd72020-05-03 22:30:49 +02002872
2873 MakeArgRefsVarargs('arg_val', 'one', 'two')
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002874 g:UseVararg('call_val')->assert_equal('arg_val/the_loc/call_val/one two')
Bram Moolenaar2fd4cd72020-05-03 22:30:49 +02002875
2876 unlet g:UseArg
2877 unlet g:UseVararg
Bram Moolenaar44ec21c2021-02-12 21:50:57 +01002878
2879 var lines =<< trim END
2880 vim9script
2881 def Test(Fun: func(number): number): list<number>
2882 return map([1, 2, 3], (_, i) => Fun(i))
2883 enddef
2884 def Inc(nr: number): number
2885 return nr + 2
2886 enddef
2887 assert_equal([3, 4, 5], Test(Inc))
2888 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002889 v9.CheckScriptSuccess(lines)
Bram Moolenaar2fd4cd72020-05-03 22:30:49 +02002890enddef
2891
Bram Moolenaar62aec932022-01-29 21:45:34 +00002892def s:MakeGetAndAppendRefs()
Bram Moolenaar85d5e2b2020-10-10 14:13:01 +02002893 var local = 'a'
2894
2895 def Append(arg: string)
2896 local ..= arg
2897 enddef
2898 g:Append = Append
2899
2900 def Get(): string
2901 return local
2902 enddef
2903 g:Get = Get
2904enddef
2905
2906def Test_closure_append_get()
2907 MakeGetAndAppendRefs()
2908 g:Get()->assert_equal('a')
2909 g:Append('-b')
2910 g:Get()->assert_equal('a-b')
2911 g:Append('-c')
2912 g:Get()->assert_equal('a-b-c')
2913
2914 unlet g:Append
2915 unlet g:Get
2916enddef
Bram Moolenaarb68b3462020-05-06 21:06:30 +02002917
Bram Moolenaar04b12692020-05-04 23:24:44 +02002918def Test_nested_closure()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002919 var local = 'text'
Bram Moolenaar04b12692020-05-04 23:24:44 +02002920 def Closure(arg: string): string
2921 return local .. arg
2922 enddef
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002923 Closure('!!!')->assert_equal('text!!!')
Bram Moolenaar04b12692020-05-04 23:24:44 +02002924enddef
2925
Bram Moolenaar62aec932022-01-29 21:45:34 +00002926func s:GetResult(Ref)
Bram Moolenaar6f5b6df2020-05-16 21:20:12 +02002927 return a:Ref('some')
2928endfunc
2929
2930def Test_call_closure_not_compiled()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002931 var text = 'text'
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01002932 g:Ref = (s) => s .. text
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002933 GetResult(g:Ref)->assert_equal('sometext')
Bram Moolenaar6f5b6df2020-05-16 21:20:12 +02002934enddef
2935
Bram Moolenaar7cbfaa52020-09-18 21:25:32 +02002936def Test_double_closure_fails()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002937 var lines =<< trim END
Bram Moolenaar7cbfaa52020-09-18 21:25:32 +02002938 vim9script
2939 def Func()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002940 var name = 0
2941 for i in range(2)
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01002942 timer_start(0, () => name)
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002943 endfor
Bram Moolenaar7cbfaa52020-09-18 21:25:32 +02002944 enddef
2945 Func()
2946 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002947 v9.CheckScriptSuccess(lines)
Bram Moolenaar7cbfaa52020-09-18 21:25:32 +02002948enddef
2949
Bram Moolenaar85d5e2b2020-10-10 14:13:01 +02002950def Test_nested_closure_used()
2951 var lines =<< trim END
2952 vim9script
2953 def Func()
2954 var x = 'hello'
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01002955 var Closure = () => x
2956 g:Myclosure = () => Closure()
Bram Moolenaar85d5e2b2020-10-10 14:13:01 +02002957 enddef
2958 Func()
2959 assert_equal('hello', g:Myclosure())
2960 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002961 v9.CheckScriptSuccess(lines)
Bram Moolenaar85d5e2b2020-10-10 14:13:01 +02002962enddef
Bram Moolenaar0876c782020-10-07 19:08:04 +02002963
Bram Moolenaarc70bdab2020-09-26 19:59:38 +02002964def Test_nested_closure_fails()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002965 var lines =<< trim END
Bram Moolenaarc70bdab2020-09-26 19:59:38 +02002966 vim9script
2967 def FuncA()
2968 FuncB(0)
2969 enddef
2970 def FuncB(n: number): list<string>
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01002971 return map([0], (_, v) => n)
Bram Moolenaarc70bdab2020-09-26 19:59:38 +02002972 enddef
2973 FuncA()
2974 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002975 v9.CheckScriptFailure(lines, 'E1012:')
Bram Moolenaarc70bdab2020-09-26 19:59:38 +02002976enddef
2977
Bram Moolenaar6de22962022-09-09 21:35:36 +01002978def Run_Test_closure_in_for_loop_fails()
2979 var lines =<< trim END
2980 vim9script
Bram Moolenaar766ae5b2022-09-14 00:30:51 +01002981 redraw
Bram Moolenaar6de22962022-09-09 21:35:36 +01002982 for n in [0]
Bram Moolenaar766ae5b2022-09-14 00:30:51 +01002983 # time should be enough for startup to finish
2984 timer_start(200, (_) => {
Bram Moolenaar6de22962022-09-09 21:35:36 +01002985 echo n
2986 })
2987 endfor
2988 END
2989 writefile(lines, 'XTest_closure_fails', 'D')
2990
2991 # Check that an error shows
Bram Moolenaarc069ede2022-09-11 12:01:04 +01002992 var buf = g:RunVimInTerminal('-S XTest_closure_fails', {rows: 6, wait_for_ruler: 0})
Bram Moolenaar766ae5b2022-09-14 00:30:51 +01002993 g:VerifyScreenDump(buf, 'Test_vim9_closure_fails', {wait: 3000})
Bram Moolenaar6de22962022-09-09 21:35:36 +01002994
2995 # clean up
2996 g:StopVimInTerminal(buf)
2997enddef
2998
2999func Test_closure_in_for_loop_fails()
3000 CheckScreendump
3001 call Run_Test_closure_in_for_loop_fails()
3002endfunc
3003
Bram Moolenaarf112f302020-12-20 17:47:52 +01003004def Test_global_closure()
3005 var lines =<< trim END
3006 vim9script
3007 def ReverseEveryNLines(n: number, line1: number, line2: number)
3008 var mods = 'sil keepj keepp lockm '
3009 var range = ':' .. line1 .. ',' .. line2
3010 def g:Offset(): number
3011 var offset = (line('.') - line1 + 1) % n
3012 return offset != 0 ? offset : n
3013 enddef
3014 exe mods .. range .. 'g/^/exe "m .-" .. g:Offset()'
3015 enddef
3016
3017 new
3018 repeat(['aaa', 'bbb', 'ccc'], 3)->setline(1)
3019 ReverseEveryNLines(3, 1, 9)
3020 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003021 v9.CheckScriptSuccess(lines)
Bram Moolenaarf112f302020-12-20 17:47:52 +01003022 var expected = repeat(['ccc', 'bbb', 'aaa'], 3)
3023 assert_equal(expected, getline(1, 9))
3024 bwipe!
3025enddef
3026
Bram Moolenaarcd45ed02020-12-22 17:35:54 +01003027def Test_global_closure_called_directly()
3028 var lines =<< trim END
3029 vim9script
3030 def Outer()
3031 var x = 1
3032 def g:Inner()
3033 var y = x
3034 x += 1
3035 assert_equal(1, y)
3036 enddef
3037 g:Inner()
3038 assert_equal(2, x)
3039 enddef
3040 Outer()
3041 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003042 v9.CheckScriptSuccess(lines)
Bram Moolenaarcd45ed02020-12-22 17:35:54 +01003043 delfunc g:Inner
3044enddef
3045
Bram Moolenaar69c76172021-12-02 16:38:52 +00003046def Test_closure_called_from_legacy()
3047 var lines =<< trim END
3048 vim9script
3049 def Func()
3050 var outer = 'foo'
3051 var F = () => {
3052 outer = 'bar'
3053 }
3054 execute printf('call %s()', string(F))
3055 enddef
3056 Func()
3057 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003058 v9.CheckScriptFailure(lines, 'E1248')
Bram Moolenaar69c76172021-12-02 16:38:52 +00003059enddef
3060
Bram Moolenaar34c54eb2020-11-25 19:15:19 +01003061def Test_failure_in_called_function()
3062 # this was using the frame index as the return value
3063 var lines =<< trim END
3064 vim9script
3065 au TerminalWinOpen * eval [][0]
3066 def PopupTerm(a: any)
3067 # make sure typvals on stack are string
3068 ['a', 'b', 'c', 'd', 'e', 'f', 'g']->join()
3069 FireEvent()
3070 enddef
3071 def FireEvent()
3072 do TerminalWinOpen
3073 enddef
3074 # use try/catch to make eval fail
3075 try
3076 call PopupTerm(0)
3077 catch
3078 endtry
3079 au! TerminalWinOpen
3080 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003081 v9.CheckScriptSuccess(lines)
Bram Moolenaar34c54eb2020-11-25 19:15:19 +01003082enddef
3083
Bram Moolenaar5366e1a2020-10-01 13:01:34 +02003084def Test_nested_lambda()
3085 var lines =<< trim END
3086 vim9script
3087 def Func()
3088 var x = 4
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01003089 var Lambda1 = () => 7
3090 var Lambda2 = () => [Lambda1(), x]
Bram Moolenaar5366e1a2020-10-01 13:01:34 +02003091 var res = Lambda2()
3092 assert_equal([7, 4], res)
3093 enddef
3094 Func()
3095 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003096 v9.CheckScriptSuccess(lines)
Bram Moolenaar5366e1a2020-10-01 13:01:34 +02003097enddef
3098
Bram Moolenaarc04f2a42021-06-09 19:30:03 +02003099def Test_double_nested_lambda()
3100 var lines =<< trim END
3101 vim9script
3102 def F(head: string): func(string): func(string): string
3103 return (sep: string): func(string): string => ((tail: string): string => {
3104 return head .. sep .. tail
3105 })
3106 enddef
3107 assert_equal('hello-there', F('hello')('-')('there'))
3108 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003109 v9.CheckScriptSuccess(lines)
Bram Moolenaarc04f2a42021-06-09 19:30:03 +02003110enddef
3111
Bram Moolenaar074f84c2021-05-18 11:47:44 +02003112def Test_nested_inline_lambda()
Bram Moolenaar074f84c2021-05-18 11:47:44 +02003113 var lines =<< trim END
3114 vim9script
3115 def F(text: string): func(string): func(string): string
3116 return (arg: string): func(string): string => ((sep: string): string => {
Bram Moolenaar23e2e112021-08-03 21:16:18 +02003117 return sep .. arg .. text
Bram Moolenaar074f84c2021-05-18 11:47:44 +02003118 })
3119 enddef
Bram Moolenaar23e2e112021-08-03 21:16:18 +02003120 assert_equal('--there++', F('++')('there')('--'))
Bram Moolenaar074f84c2021-05-18 11:47:44 +02003121 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003122 v9.CheckScriptSuccess(lines)
Bram Moolenaar5245beb2021-07-15 22:03:50 +02003123
3124 lines =<< trim END
3125 vim9script
3126 echo range(4)->mapnew((_, v) => {
3127 return range(v) ->mapnew((_, s) => {
3128 return string(s)
3129 })
3130 })
3131 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003132 v9.CheckScriptSuccess(lines)
Bram Moolenaarc6ba2f92021-07-18 13:42:29 +02003133
3134 lines =<< trim END
3135 vim9script
3136
Bram Moolenaara749a422022-02-12 19:52:25 +00003137 def Func()
Bram Moolenaarc6ba2f92021-07-18 13:42:29 +02003138 range(10)
3139 ->mapnew((_, _) => ({
3140 key: range(10)->mapnew((_, _) => {
3141 return ' '
3142 }),
3143 }))
3144 enddef
3145
3146 defcomp
3147 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003148 v9.CheckScriptSuccess(lines)
Bram Moolenaar074f84c2021-05-18 11:47:44 +02003149enddef
3150
Bram Moolenaar52bf81c2020-11-17 18:50:44 +01003151def Shadowed(): list<number>
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01003152 var FuncList: list<func: number> = [() => 42]
Bram Moolenaar75ab91f2021-01-10 22:42:50 +01003153 return FuncList->mapnew((_, Shadowed) => Shadowed())
Bram Moolenaar52bf81c2020-11-17 18:50:44 +01003154enddef
3155
3156def Test_lambda_arg_shadows_func()
Bram Moolenaar62aec932022-01-29 21:45:34 +00003157 assert_equal([42], g:Shadowed())
Bram Moolenaar52bf81c2020-11-17 18:50:44 +01003158enddef
3159
Bram Moolenaar21dc8f12022-03-16 17:54:17 +00003160def Test_compiling_referenced_func_no_shadow()
3161 var lines =<< trim END
3162 vim9script
3163
3164 def InitializeReply(lspserver: dict<any>)
3165 enddef
3166
3167 def ProcessReply(lspserver: dict<any>)
3168 var lsp_reply_handlers: dict<func> =
3169 { 'initialize': InitializeReply }
3170 lsp_reply_handlers['initialize'](lspserver)
3171 enddef
3172
3173 call ProcessReply({})
3174 END
3175 v9.CheckScriptSuccess(lines)
3176enddef
3177
Bram Moolenaar62aec932022-01-29 21:45:34 +00003178def s:Line_continuation_in_def(dir: string = ''): string
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02003179 var path: string = empty(dir)
3180 \ ? 'empty'
3181 \ : 'full'
3182 return path
Bram Moolenaaracd4c5e2020-06-22 19:39:03 +02003183enddef
3184
3185def Test_line_continuation_in_def()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02003186 Line_continuation_in_def('.')->assert_equal('full')
Bram Moolenaaracd4c5e2020-06-22 19:39:03 +02003187enddef
3188
Bram Moolenaar2ea95b62020-11-19 21:47:56 +01003189def Test_script_var_in_lambda()
3190 var lines =<< trim END
3191 vim9script
3192 var script = 'test'
Bram Moolenaarbb8a7ce2021-04-10 20:10:26 +02003193 assert_equal(['test'], map(['one'], (_, _) => script))
Bram Moolenaar2ea95b62020-11-19 21:47:56 +01003194 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003195 v9.CheckScriptSuccess(lines)
Bram Moolenaar2ea95b62020-11-19 21:47:56 +01003196enddef
3197
Bram Moolenaar62aec932022-01-29 21:45:34 +00003198def s:Line_continuation_in_lambda(): list<string>
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02003199 var x = range(97, 100)
Bram Moolenaar75ab91f2021-01-10 22:42:50 +01003200 ->mapnew((_, v) => nr2char(v)
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01003201 ->toupper())
Bram Moolenaar7a4b8982020-07-08 17:36:21 +02003202 ->reverse()
3203 return x
3204enddef
3205
3206def Test_line_continuation_in_lambda()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02003207 Line_continuation_in_lambda()->assert_equal(['D', 'C', 'B', 'A'])
Bram Moolenaarf898f7c2021-01-16 18:09:52 +01003208
3209 var lines =<< trim END
3210 vim9script
3211 var res = [{n: 1, m: 2, s: 'xxx'}]
3212 ->mapnew((_, v: dict<any>): string => printf('%d:%d:%s',
3213 v.n,
3214 v.m,
3215 substitute(v.s, '.*', 'yyy', '')
3216 ))
3217 assert_equal(['1:2:yyy'], res)
3218 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003219 v9.CheckScriptSuccess(lines)
Bram Moolenaar7a4b8982020-07-08 17:36:21 +02003220enddef
3221
Bram Moolenaarb6571982021-01-08 22:24:19 +01003222def Test_list_lambda()
3223 timer_start(1000, (_) => 0)
3224 var body = execute(timer_info()[0].callback
3225 ->string()
3226 ->substitute("('", ' ', '')
3227 ->substitute("')", '', '')
3228 ->substitute('function\zs', ' ', ''))
Bram Moolenaar767034c2021-04-09 17:24:52 +02003229 assert_match('def <lambda>\d\+(_: any): number\n1 return 0\n enddef', body)
Bram Moolenaarb6571982021-01-08 22:24:19 +01003230enddef
3231
Bram Moolenaar3c77b6a2021-07-25 18:07:00 +02003232def Test_lambda_block_variable()
Bram Moolenaar88421d62021-07-24 14:14:52 +02003233 var lines =<< trim END
3234 vim9script
3235 var flist: list<func>
3236 for i in range(10)
3237 var inloop = i
3238 flist[i] = () => inloop
3239 endfor
3240 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003241 v9.CheckScriptSuccess(lines)
Bram Moolenaar88421d62021-07-24 14:14:52 +02003242
3243 lines =<< trim END
3244 vim9script
3245 if true
3246 var outloop = 5
3247 var flist: list<func>
3248 for i in range(10)
3249 flist[i] = () => outloop
3250 endfor
3251 endif
3252 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003253 v9.CheckScriptSuccess(lines)
Bram Moolenaar88421d62021-07-24 14:14:52 +02003254
3255 lines =<< trim END
3256 vim9script
3257 if true
3258 var outloop = 5
3259 endif
3260 var flist: list<func>
3261 for i in range(10)
3262 flist[i] = () => outloop
3263 endfor
3264 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003265 v9.CheckScriptFailure(lines, 'E1001: Variable not found: outloop', 1)
Bram Moolenaar3c77b6a2021-07-25 18:07:00 +02003266
3267 lines =<< trim END
3268 vim9script
3269 for i in range(10)
3270 var Ref = () => 0
3271 endfor
3272 assert_equal(0, ((i) => 0)(0))
3273 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003274 v9.CheckScriptSuccess(lines)
Bram Moolenaar88421d62021-07-24 14:14:52 +02003275enddef
3276
Bram Moolenaar96cf4ba2021-04-24 14:15:41 +02003277def Test_legacy_lambda()
3278 legacy echo {x -> 'hello ' .. x}('foo')
Bram Moolenaardc4c2302021-04-25 13:54:42 +02003279
Bram Moolenaar96cf4ba2021-04-24 14:15:41 +02003280 var lines =<< trim END
3281 echo {x -> 'hello ' .. x}('foo')
3282 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003283 v9.CheckDefAndScriptFailure(lines, 'E720:')
Bram Moolenaardc4c2302021-04-25 13:54:42 +02003284
3285 lines =<< trim END
3286 vim9script
3287 def Func()
3288 echo (() => 'no error')()
3289 enddef
3290 legacy call s:Func()
3291 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003292 v9.CheckScriptSuccess(lines)
Bram Moolenaar96cf4ba2021-04-24 14:15:41 +02003293enddef
3294
Bram Moolenaarce024c32021-06-26 13:00:49 +02003295def Test_legacy()
3296 var lines =<< trim END
3297 vim9script
3298 func g:LegacyFunction()
3299 let g:legacyvar = 1
3300 endfunc
3301 def Testit()
3302 legacy call g:LegacyFunction()
3303 enddef
3304 Testit()
3305 assert_equal(1, g:legacyvar)
3306 unlet g:legacyvar
3307 delfunc g:LegacyFunction
3308 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003309 v9.CheckScriptSuccess(lines)
Bram Moolenaarce024c32021-06-26 13:00:49 +02003310enddef
3311
Bram Moolenaarc3cb1c92021-06-02 16:47:53 +02003312def Test_legacy_errors()
3313 for cmd in ['if', 'elseif', 'else', 'endif',
3314 'for', 'endfor', 'continue', 'break',
3315 'while', 'endwhile',
3316 'try', 'catch', 'finally', 'endtry']
Bram Moolenaar62aec932022-01-29 21:45:34 +00003317 v9.CheckDefFailure(['legacy ' .. cmd .. ' expr'], 'E1189:')
Bram Moolenaarc3cb1c92021-06-02 16:47:53 +02003318 endfor
3319enddef
3320
Bram Moolenaarb1b6f4d2021-09-13 18:25:54 +02003321def Test_call_legacy_with_dict()
3322 var lines =<< trim END
3323 vim9script
3324 func Legacy() dict
3325 let g:result = self.value
3326 endfunc
3327 def TestDirect()
3328 var d = {value: 'yes', func: Legacy}
3329 d.func()
3330 enddef
3331 TestDirect()
3332 assert_equal('yes', g:result)
3333 unlet g:result
3334
3335 def TestIndirect()
3336 var d = {value: 'foo', func: Legacy}
3337 var Fi = d.func
3338 Fi()
3339 enddef
3340 TestIndirect()
3341 assert_equal('foo', g:result)
3342 unlet g:result
3343
3344 var d = {value: 'bar', func: Legacy}
3345 d.func()
3346 assert_equal('bar', g:result)
3347 unlet g:result
3348 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003349 v9.CheckScriptSuccess(lines)
Bram Moolenaarb1b6f4d2021-09-13 18:25:54 +02003350enddef
3351
Bram Moolenaar62aec932022-01-29 21:45:34 +00003352def s:DoFilterThis(a: string): list<string>
Bram Moolenaarab360522021-01-10 14:02:28 +01003353 # closure nested inside another closure using argument
3354 var Filter = (l) => filter(l, (_, v) => stridx(v, a) == 0)
3355 return ['x', 'y', 'a', 'x2', 'c']->Filter()
3356enddef
3357
3358def Test_nested_closure_using_argument()
3359 assert_equal(['x', 'x2'], DoFilterThis('x'))
3360enddef
3361
Bram Moolenaar0186e582021-01-10 18:33:11 +01003362def Test_triple_nested_closure()
3363 var what = 'x'
3364 var Match = (val: string, cmp: string): bool => stridx(val, cmp) == 0
3365 var Filter = (l) => filter(l, (_, v) => Match(v, what))
3366 assert_equal(['x', 'x2'], ['x', 'y', 'a', 'x2', 'c']->Filter())
3367enddef
3368
Bram Moolenaar8f510af2020-07-05 18:48:23 +02003369func Test_silent_echo()
Bram Moolenaar47e7d702020-07-05 18:18:42 +02003370 CheckScreendump
Bram Moolenaar3b309f12021-12-13 18:19:55 +00003371 call Run_Test_silent_echo()
3372endfunc
Bram Moolenaar47e7d702020-07-05 18:18:42 +02003373
Bram Moolenaar3b309f12021-12-13 18:19:55 +00003374def Run_Test_silent_echo()
3375 var lines =<< trim END
Bram Moolenaar47e7d702020-07-05 18:18:42 +02003376 vim9script
3377 def EchoNothing()
3378 silent echo ''
3379 enddef
3380 defcompile
3381 END
Bram Moolenaar6de22962022-09-09 21:35:36 +01003382 writefile(lines, 'XTest_silent_echo', 'D')
Bram Moolenaar47e7d702020-07-05 18:18:42 +02003383
Bram Moolenaar3b309f12021-12-13 18:19:55 +00003384 # Check that the balloon shows up after a mouse move
Bram Moolenaar62aec932022-01-29 21:45:34 +00003385 var buf = g:RunVimInTerminal('-S XTest_silent_echo', {'rows': 6})
Bram Moolenaar3b309f12021-12-13 18:19:55 +00003386 term_sendkeys(buf, ":abc")
Bram Moolenaar62aec932022-01-29 21:45:34 +00003387 g:VerifyScreenDump(buf, 'Test_vim9_silent_echo', {})
Bram Moolenaar47e7d702020-07-05 18:18:42 +02003388
Bram Moolenaar3b309f12021-12-13 18:19:55 +00003389 # clean up
Bram Moolenaar62aec932022-01-29 21:45:34 +00003390 g:StopVimInTerminal(buf)
Bram Moolenaar3b309f12021-12-13 18:19:55 +00003391enddef
Bram Moolenaar47e7d702020-07-05 18:18:42 +02003392
Bram Moolenaar171fb922020-10-28 16:54:47 +01003393def SilentlyError()
3394 execute('silent! invalid')
3395 g:did_it = 'yes'
3396enddef
3397
Bram Moolenaar62aec932022-01-29 21:45:34 +00003398func s:UserError()
Bram Moolenaar28ee8922020-10-28 20:20:00 +01003399 silent! invalid
3400endfunc
3401
3402def SilentlyUserError()
3403 UserError()
3404 g:did_it = 'yes'
3405enddef
Bram Moolenaar171fb922020-10-28 16:54:47 +01003406
3407" This can't be a :def function, because the assert would not be reached.
Bram Moolenaar171fb922020-10-28 16:54:47 +01003408func Test_ignore_silent_error()
3409 let g:did_it = 'no'
3410 call SilentlyError()
3411 call assert_equal('yes', g:did_it)
3412
Bram Moolenaar28ee8922020-10-28 20:20:00 +01003413 let g:did_it = 'no'
3414 call SilentlyUserError()
3415 call assert_equal('yes', g:did_it)
Bram Moolenaar171fb922020-10-28 16:54:47 +01003416
3417 unlet g:did_it
3418endfunc
3419
Bram Moolenaarcd030c42020-10-30 21:49:40 +01003420def Test_ignore_silent_error_in_filter()
3421 var lines =<< trim END
3422 vim9script
3423 def Filter(winid: number, key: string): bool
3424 if key == 'o'
3425 silent! eval [][0]
3426 return true
3427 endif
3428 return popup_filter_menu(winid, key)
3429 enddef
3430
Bram Moolenaare0de1712020-12-02 17:36:54 +01003431 popup_create('popup', {filter: Filter})
Bram Moolenaarcd030c42020-10-30 21:49:40 +01003432 feedkeys("o\r", 'xnt')
3433 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003434 v9.CheckScriptSuccess(lines)
Bram Moolenaarcd030c42020-10-30 21:49:40 +01003435enddef
3436
Bram Moolenaar62aec932022-01-29 21:45:34 +00003437def s:Fibonacci(n: number): number
Bram Moolenaar4b9bd692020-09-05 21:57:53 +02003438 if n < 2
3439 return n
3440 else
3441 return Fibonacci(n - 1) + Fibonacci(n - 2)
3442 endif
3443enddef
3444
Bram Moolenaar985116a2020-07-12 17:31:09 +02003445def Test_recursive_call()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02003446 Fibonacci(20)->assert_equal(6765)
Bram Moolenaar985116a2020-07-12 17:31:09 +02003447enddef
3448
Bram Moolenaar62aec932022-01-29 21:45:34 +00003449def s:TreeWalk(dir: string): list<any>
Bram Moolenaar75ab91f2021-01-10 22:42:50 +01003450 return readdir(dir)->mapnew((_, val) =>
Bram Moolenaar08f7a412020-07-13 20:41:08 +02003451 fnamemodify(dir .. '/' .. val, ':p')->isdirectory()
Bram Moolenaar2bede172020-11-19 18:53:18 +01003452 ? {[val]: TreeWalk(dir .. '/' .. val)}
Bram Moolenaar08f7a412020-07-13 20:41:08 +02003453 : val
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01003454 )
Bram Moolenaar08f7a412020-07-13 20:41:08 +02003455enddef
3456
3457def Test_closure_in_map()
Bram Moolenaarf5fec052022-09-11 11:49:22 +01003458 mkdir('XclosureDir/tdir', 'pR')
Bram Moolenaar08f7a412020-07-13 20:41:08 +02003459 writefile(['111'], 'XclosureDir/file1')
3460 writefile(['222'], 'XclosureDir/file2')
3461 writefile(['333'], 'XclosureDir/tdir/file3')
3462
Bram Moolenaare0de1712020-12-02 17:36:54 +01003463 TreeWalk('XclosureDir')->assert_equal(['file1', 'file2', {tdir: ['file3']}])
Bram Moolenaar08f7a412020-07-13 20:41:08 +02003464enddef
3465
Bram Moolenaar7b5d5442020-10-04 13:42:34 +02003466def Test_invalid_function_name()
3467 var lines =<< trim END
3468 vim9script
3469 def s: list<string>
3470 END
Bram Moolenaara749a422022-02-12 19:52:25 +00003471 v9.CheckScriptFailure(lines, 'E1268:')
Bram Moolenaar7b5d5442020-10-04 13:42:34 +02003472
3473 lines =<< trim END
3474 vim9script
3475 def g: list<string>
3476 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003477 v9.CheckScriptFailure(lines, 'E129:')
Bram Moolenaar7b5d5442020-10-04 13:42:34 +02003478
3479 lines =<< trim END
3480 vim9script
3481 def <SID>: list<string>
3482 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003483 v9.CheckScriptFailure(lines, 'E884:')
Bram Moolenaar7b5d5442020-10-04 13:42:34 +02003484
3485 lines =<< trim END
3486 vim9script
3487 def F list<string>
3488 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003489 v9.CheckScriptFailure(lines, 'E488:')
Bram Moolenaar7b5d5442020-10-04 13:42:34 +02003490enddef
3491
Bram Moolenaara90afb92020-07-15 22:38:56 +02003492def Test_partial_call()
Bram Moolenaarf78da4f2021-08-01 15:40:31 +02003493 var lines =<< trim END
3494 var Xsetlist: func
3495 Xsetlist = function('setloclist', [0])
3496 Xsetlist([], ' ', {title: 'test'})
3497 getloclist(0, {title: 1})->assert_equal({title: 'test'})
Bram Moolenaara90afb92020-07-15 22:38:56 +02003498
Bram Moolenaarf78da4f2021-08-01 15:40:31 +02003499 Xsetlist = function('setloclist', [0, [], ' '])
3500 Xsetlist({title: 'test'})
3501 getloclist(0, {title: 1})->assert_equal({title: 'test'})
Bram Moolenaara90afb92020-07-15 22:38:56 +02003502
Bram Moolenaarf78da4f2021-08-01 15:40:31 +02003503 Xsetlist = function('setqflist')
3504 Xsetlist([], ' ', {title: 'test'})
3505 getqflist({title: 1})->assert_equal({title: 'test'})
Bram Moolenaara90afb92020-07-15 22:38:56 +02003506
Bram Moolenaarf78da4f2021-08-01 15:40:31 +02003507 Xsetlist = function('setqflist', [[], ' '])
3508 Xsetlist({title: 'test'})
3509 getqflist({title: 1})->assert_equal({title: 'test'})
Bram Moolenaar6abd3dc2020-10-04 14:17:32 +02003510
Bram Moolenaarf78da4f2021-08-01 15:40:31 +02003511 var Len: func: number = function('len', ['word'])
3512 assert_equal(4, Len())
3513
3514 var RepeatFunc = function('repeat', ['o'])
3515 assert_equal('ooooo', RepeatFunc(5))
3516 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003517 v9.CheckDefAndScriptSuccess(lines)
Bram Moolenaarc66f6452021-08-19 21:08:30 +02003518
3519 lines =<< trim END
3520 vim9script
3521 def Foo(Parser: any)
3522 enddef
3523 var Expr: func(dict<any>): dict<any>
3524 const Call = Foo(Expr)
3525 END
Bram Moolenaar8acb9cc2022-03-08 13:18:55 +00003526 v9.CheckScriptFailure(lines, 'E1031:')
Bram Moolenaara90afb92020-07-15 22:38:56 +02003527enddef
3528
Bram Moolenaarcd1cda22022-02-16 21:48:25 +00003529def Test_partial_double_nested()
3530 var idx = 123
3531 var Get = () => idx
3532 var Ref = function(Get, [])
3533 var RefRef = function(Ref, [])
3534 assert_equal(123, RefRef())
3535enddef
3536
Bram Moolenaar673bcb12022-03-08 16:52:24 +00003537def Test_partial_null_function()
3538 var lines =<< trim END
3539 var d: dict<func> = {f: null_function}
3540 var Ref = d.f
Bram Moolenaared0c62e2022-03-08 19:43:55 +00003541 assert_equal('func(...): unknown', typename(Ref))
Bram Moolenaar673bcb12022-03-08 16:52:24 +00003542 END
3543 v9.CheckDefAndScriptSuccess(lines)
3544enddef
3545
Bram Moolenaar2dd0a2c2020-08-08 15:10:27 +02003546def Test_cmd_modifier()
3547 tab echo '0'
Bram Moolenaar62aec932022-01-29 21:45:34 +00003548 v9.CheckDefFailure(['5tab echo 3'], 'E16:')
Bram Moolenaar2dd0a2c2020-08-08 15:10:27 +02003549enddef
3550
3551def Test_restore_modifiers()
3552 # check that when compiling a :def function command modifiers are not messed
3553 # up.
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02003554 var lines =<< trim END
Bram Moolenaar2dd0a2c2020-08-08 15:10:27 +02003555 vim9script
3556 set eventignore=
3557 autocmd QuickFixCmdPost * copen
3558 def AutocmdsDisabled()
Bram Moolenaarc3235272021-07-10 19:42:03 +02003559 eval 1 + 2
Bram Moolenaar2dd0a2c2020-08-08 15:10:27 +02003560 enddef
3561 func Func()
3562 noautocmd call s:AutocmdsDisabled()
3563 let g:ei_after = &eventignore
3564 endfunc
3565 Func()
3566 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003567 v9.CheckScriptSuccess(lines)
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02003568 g:ei_after->assert_equal('')
Bram Moolenaar2dd0a2c2020-08-08 15:10:27 +02003569enddef
3570
Bram Moolenaardfa3d552020-09-10 22:05:08 +02003571def StackTop()
Bram Moolenaarc3235272021-07-10 19:42:03 +02003572 eval 1 + 2
3573 eval 2 + 3
Bram Moolenaardfa3d552020-09-10 22:05:08 +02003574 # call not on fourth line
Bram Moolenaar62aec932022-01-29 21:45:34 +00003575 g:StackBot()
Bram Moolenaardfa3d552020-09-10 22:05:08 +02003576enddef
3577
3578def StackBot()
3579 # throw an error
3580 eval [][0]
3581enddef
3582
3583def Test_callstack_def()
3584 try
Bram Moolenaar62aec932022-01-29 21:45:34 +00003585 g:StackTop()
Bram Moolenaardfa3d552020-09-10 22:05:08 +02003586 catch
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02003587 v:throwpoint->assert_match('Test_callstack_def\[2\]..StackTop\[4\]..StackBot, line 2')
Bram Moolenaardfa3d552020-09-10 22:05:08 +02003588 endtry
3589enddef
3590
Bram Moolenaare8211a32020-10-09 22:04:29 +02003591" Re-using spot for variable used in block
3592def Test_block_scoped_var()
3593 var lines =<< trim END
3594 vim9script
3595 def Func()
3596 var x = ['a', 'b', 'c']
3597 if 1
3598 var y = 'x'
Bram Moolenaarbb8a7ce2021-04-10 20:10:26 +02003599 map(x, (_, _) => y)
Bram Moolenaare8211a32020-10-09 22:04:29 +02003600 endif
3601 var z = x
3602 assert_equal(['x', 'x', 'x'], z)
3603 enddef
3604 Func()
3605 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003606 v9.CheckScriptSuccess(lines)
Bram Moolenaare8211a32020-10-09 22:04:29 +02003607enddef
3608
Bram Moolenaareeece9e2020-11-20 19:26:48 +01003609def Test_reset_did_emsg()
3610 var lines =<< trim END
3611 @s = 'blah'
3612 au BufWinLeave * #
3613 def Func()
3614 var winid = popup_create('popup', {})
3615 exe '*s'
3616 popup_close(winid)
3617 enddef
3618 Func()
3619 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003620 v9.CheckScriptFailure(lines, 'E492:', 8)
Bram Moolenaar2d870f82020-12-05 13:41:01 +01003621 delfunc! g:Func
Bram Moolenaareeece9e2020-11-20 19:26:48 +01003622enddef
3623
Bram Moolenaar57f799e2020-12-12 20:42:19 +01003624def Test_did_emsg_reset()
3625 # executing an autocommand resets did_emsg, this should not result in a
3626 # builtin function considered failing
3627 var lines =<< trim END
3628 vim9script
3629 au BufWinLeave * #
3630 def Func()
Bram Moolenaar767034c2021-04-09 17:24:52 +02003631 popup_menu('', {callback: (a, b) => popup_create('', {})->popup_close()})
Bram Moolenaar57f799e2020-12-12 20:42:19 +01003632 eval [][0]
3633 enddef
3634 nno <F3> <cmd>call <sid>Func()<cr>
3635 feedkeys("\<F3>\e", 'xt')
3636 END
Bram Moolenaarf5fec052022-09-11 11:49:22 +01003637 writefile(lines, 'XemsgReset', 'D')
Bram Moolenaar57f799e2020-12-12 20:42:19 +01003638 assert_fails('so XemsgReset', ['E684:', 'E684:'], lines, 2)
Bram Moolenaarf5fec052022-09-11 11:49:22 +01003639
Bram Moolenaar57f799e2020-12-12 20:42:19 +01003640 nunmap <F3>
3641 au! BufWinLeave
3642enddef
3643
Bram Moolenaar56602ba2020-12-05 21:22:08 +01003644def Test_abort_with_silent_call()
3645 var lines =<< trim END
3646 vim9script
3647 g:result = 'none'
3648 def Func()
3649 g:result += 3
3650 g:result = 'yes'
3651 enddef
3652 # error is silenced, but function aborts on error
3653 silent! Func()
3654 assert_equal('none', g:result)
3655 unlet g:result
3656 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003657 v9.CheckScriptSuccess(lines)
Bram Moolenaar56602ba2020-12-05 21:22:08 +01003658enddef
3659
Bram Moolenaarf665e972020-12-05 19:17:16 +01003660def Test_continues_with_silent_error()
3661 var lines =<< trim END
3662 vim9script
3663 g:result = 'none'
3664 def Func()
3665 silent! g:result += 3
3666 g:result = 'yes'
3667 enddef
3668 # error is silenced, function does not abort
3669 Func()
3670 assert_equal('yes', g:result)
3671 unlet g:result
3672 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003673 v9.CheckScriptSuccess(lines)
Bram Moolenaarf665e972020-12-05 19:17:16 +01003674enddef
3675
Bram Moolenaaraf0df472020-12-02 20:51:22 +01003676def Test_abort_even_with_silent()
3677 var lines =<< trim END
3678 vim9script
3679 g:result = 'none'
3680 def Func()
3681 eval {-> ''}() .. '' .. {}['X']
3682 g:result = 'yes'
3683 enddef
Bram Moolenaarf665e972020-12-05 19:17:16 +01003684 silent! Func()
Bram Moolenaaraf0df472020-12-02 20:51:22 +01003685 assert_equal('none', g:result)
Bram Moolenaar4029cab2020-12-05 18:13:27 +01003686 unlet g:result
3687 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003688 v9.CheckScriptSuccess(lines)
Bram Moolenaar4029cab2020-12-05 18:13:27 +01003689enddef
3690
Bram Moolenaarf665e972020-12-05 19:17:16 +01003691def Test_cmdmod_silent_restored()
3692 var lines =<< trim END
3693 vim9script
3694 def Func()
3695 g:result = 'none'
3696 silent! g:result += 3
3697 g:result = 'none'
3698 g:result += 3
3699 enddef
3700 Func()
3701 END
3702 # can't use CheckScriptFailure, it ignores the :silent!
3703 var fname = 'Xdefsilent'
Bram Moolenaarf5fec052022-09-11 11:49:22 +01003704 writefile(lines, fname, 'D')
Bram Moolenaarf665e972020-12-05 19:17:16 +01003705 var caught = 'no'
3706 try
3707 exe 'source ' .. fname
3708 catch /E1030:/
3709 caught = 'yes'
3710 assert_match('Func, line 4', v:throwpoint)
3711 endtry
3712 assert_equal('yes', caught)
Bram Moolenaarf665e972020-12-05 19:17:16 +01003713enddef
3714
Bram Moolenaar2fecb532021-03-24 22:00:56 +01003715def Test_cmdmod_silent_nested()
3716 var lines =<< trim END
3717 vim9script
3718 var result = ''
3719
3720 def Error()
3721 result ..= 'Eb'
3722 eval [][0]
3723 result ..= 'Ea'
3724 enddef
3725
3726 def Crash()
3727 result ..= 'Cb'
3728 sil! Error()
3729 result ..= 'Ca'
3730 enddef
3731
3732 Crash()
3733 assert_equal('CbEbEaCa', result)
3734 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003735 v9.CheckScriptSuccess(lines)
Bram Moolenaar2fecb532021-03-24 22:00:56 +01003736enddef
3737
Bram Moolenaar4029cab2020-12-05 18:13:27 +01003738def Test_dict_member_with_silent()
3739 var lines =<< trim END
3740 vim9script
3741 g:result = 'none'
3742 var d: dict<any>
3743 def Func()
3744 try
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01003745 g:result = map([], (_, v) => ({}[v]))->join() .. d['']
Bram Moolenaar4029cab2020-12-05 18:13:27 +01003746 catch
3747 endtry
3748 enddef
3749 silent! Func()
3750 assert_equal('0', g:result)
3751 unlet g:result
Bram Moolenaaraf0df472020-12-02 20:51:22 +01003752 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003753 v9.CheckScriptSuccess(lines)
Bram Moolenaaraf0df472020-12-02 20:51:22 +01003754enddef
3755
Bram Moolenaarf9041332021-01-21 19:41:16 +01003756def Test_skip_cmds_with_silent()
3757 var lines =<< trim END
3758 vim9script
3759
3760 def Func(b: bool)
3761 Crash()
3762 enddef
3763
3764 def Crash()
3765 sil! :/not found/d _
3766 sil! :/not found/put _
3767 enddef
3768
3769 Func(true)
3770 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003771 v9.CheckScriptSuccess(lines)
Bram Moolenaarf9041332021-01-21 19:41:16 +01003772enddef
3773
Bram Moolenaar5b3d1bb2020-12-22 12:20:08 +01003774def Test_opfunc()
Bram Moolenaar848fadd2022-01-30 15:28:30 +00003775 nnoremap <F3> <cmd>set opfunc=g:Opfunc<cr>g@
Bram Moolenaar5b3d1bb2020-12-22 12:20:08 +01003776 def g:Opfunc(_: any): string
3777 setline(1, 'ASDF')
3778 return ''
3779 enddef
3780 new
3781 setline(1, 'asdf')
3782 feedkeys("\<F3>$", 'x')
3783 assert_equal('ASDF', getline(1))
3784
3785 bwipe!
3786 nunmap <F3>
3787enddef
3788
Bram Moolenaar3b309f12021-12-13 18:19:55 +00003789func Test_opfunc_error()
3790 CheckScreendump
3791 call Run_Test_opfunc_error()
3792endfunc
3793
3794def Run_Test_opfunc_error()
3795 # test that the error from Opfunc() is displayed right away
3796 var lines =<< trim END
3797 vim9script
3798
3799 def Opfunc(type: string)
3800 try
3801 eval [][0]
3802 catch /nothing/ # error not caught
3803 endtry
3804 enddef
3805 &operatorfunc = Opfunc
3806 nnoremap <expr> l <SID>L()
3807 def L(): string
3808 return 'l'
3809 enddef
3810 'x'->repeat(10)->setline(1)
3811 feedkeys('g@l', 'n')
3812 feedkeys('llll')
3813 END
Bram Moolenaarf5fec052022-09-11 11:49:22 +01003814 call writefile(lines, 'XTest_opfunc_error', 'D')
Bram Moolenaar3b309f12021-12-13 18:19:55 +00003815
Bram Moolenaar62aec932022-01-29 21:45:34 +00003816 var buf = g:RunVimInTerminal('-S XTest_opfunc_error', {rows: 6, wait_for_ruler: 0})
3817 g:WaitForAssert(() => assert_match('Press ENTER', term_getline(buf, 6)))
Bram Moolenaarec892232022-05-06 17:53:06 +01003818 g:WaitForAssert(() => assert_match('E684: List index out of range: 0', term_getline(buf, 5)))
Bram Moolenaar3b309f12021-12-13 18:19:55 +00003819
3820 # clean up
Bram Moolenaar62aec932022-01-29 21:45:34 +00003821 g:StopVimInTerminal(buf)
Bram Moolenaar3b309f12021-12-13 18:19:55 +00003822enddef
3823
Bram Moolenaar077a4232020-12-22 18:33:27 +01003824" this was crashing on exit
3825def Test_nested_lambda_in_closure()
3826 var lines =<< trim END
3827 vim9script
Bram Moolenaar227c58a2021-04-28 20:40:44 +02003828 command WriteDone writefile(['Done'], 'XnestedDone')
Bram Moolenaar077a4232020-12-22 18:33:27 +01003829 def Outer()
3830 def g:Inner()
3831 echo map([1, 2, 3], {_, v -> v + 1})
3832 enddef
3833 g:Inner()
3834 enddef
3835 defcompile
Bram Moolenaar227c58a2021-04-28 20:40:44 +02003836 # not reached
Bram Moolenaar077a4232020-12-22 18:33:27 +01003837 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003838 if !g:RunVim([], lines, '--clean -c WriteDone -c quit')
Bram Moolenaar077a4232020-12-22 18:33:27 +01003839 return
3840 endif
3841 assert_equal(['Done'], readfile('XnestedDone'))
3842 delete('XnestedDone')
3843enddef
3844
Bram Moolenaar92368aa2022-02-07 17:50:39 +00003845def Test_nested_closure_funcref()
3846 var lines =<< trim END
3847 vim9script
3848 def Func()
3849 var n: number
3850 def Nested()
3851 ++n
3852 enddef
3853 Nested()
3854 g:result_one = n
3855 var Ref = function(Nested)
3856 Ref()
3857 g:result_two = n
3858 enddef
3859 Func()
3860 END
3861 v9.CheckScriptSuccess(lines)
3862 assert_equal(1, g:result_one)
3863 assert_equal(2, g:result_two)
3864 unlet g:result_one g:result_two
3865enddef
3866
Bram Moolenaar7aca5ca2022-02-07 19:56:43 +00003867def Test_nested_closure_in_dict()
3868 var lines =<< trim END
3869 vim9script
3870 def Func(): dict<any>
3871 var n: number
3872 def Inc(): number
3873 ++n
3874 return n
3875 enddef
3876 return {inc: function(Inc)}
3877 enddef
3878 disas Func
3879 var d = Func()
3880 assert_equal(1, d.inc())
3881 assert_equal(2, d.inc())
3882 END
3883 v9.CheckScriptSuccess(lines)
3884enddef
3885
Bram Moolenaarfb43cfc2022-03-11 18:54:17 +00003886def Test_script_local_other_script()
3887 var lines =<< trim END
3888 function LegacyJob()
3889 let FuncRef = function('s:close_cb')
3890 endfunction
3891 function s:close_cb(...)
3892 endfunction
3893 END
Bram Moolenaarf5fec052022-09-11 11:49:22 +01003894 lines->writefile('Xlegacy.vim', 'D')
Bram Moolenaarfb43cfc2022-03-11 18:54:17 +00003895 source Xlegacy.vim
3896 g:LegacyJob()
3897 g:LegacyJob()
3898 g:LegacyJob()
3899
3900 delfunc g:LegacyJob
Bram Moolenaarfb43cfc2022-03-11 18:54:17 +00003901enddef
3902
Bram Moolenaar04947cc2021-03-06 19:26:46 +01003903def Test_check_func_arg_types()
3904 var lines =<< trim END
3905 vim9script
3906 def F1(x: string): string
3907 return x
3908 enddef
3909
3910 def F2(x: number): number
3911 return x + 1
3912 enddef
3913
Bram Moolenaar1d9cef72022-03-17 16:30:03 +00003914 def G(Fg: func): dict<func>
3915 return {f: Fg}
Bram Moolenaar04947cc2021-03-06 19:26:46 +01003916 enddef
3917
3918 def H(d: dict<func>): string
3919 return d.f('a')
3920 enddef
3921 END
3922
Bram Moolenaar62aec932022-01-29 21:45:34 +00003923 v9.CheckScriptSuccess(lines + ['echo H(G(F1))'])
3924 v9.CheckScriptFailure(lines + ['echo H(G(F2))'], 'E1013:')
Bram Moolenaar1d9cef72022-03-17 16:30:03 +00003925
3926 v9.CheckScriptFailure(lines + ['def SomeFunc(ff: func)', 'enddef'], 'E704:')
Bram Moolenaar04947cc2021-03-06 19:26:46 +01003927enddef
3928
Bram Moolenaarbadf04f2022-03-12 21:28:22 +00003929def Test_call_func_with_null()
3930 var lines =<< trim END
3931 def Fstring(v: string)
3932 assert_equal(null_string, v)
3933 enddef
3934 Fstring(null_string)
3935 def Fblob(v: blob)
3936 assert_equal(null_blob, v)
3937 enddef
3938 Fblob(null_blob)
3939 def Flist(v: list<number>)
3940 assert_equal(null_list, v)
3941 enddef
3942 Flist(null_list)
3943 def Fdict(v: dict<number>)
3944 assert_equal(null_dict, v)
3945 enddef
3946 Fdict(null_dict)
3947 def Ffunc(Fv: func(number): number)
3948 assert_equal(null_function, Fv)
3949 enddef
3950 Ffunc(null_function)
3951 if has('channel')
3952 def Fchannel(v: channel)
3953 assert_equal(null_channel, v)
3954 enddef
3955 Fchannel(null_channel)
3956 def Fjob(v: job)
3957 assert_equal(null_job, v)
3958 enddef
3959 Fjob(null_job)
3960 endif
3961 END
3962 v9.CheckDefAndScriptSuccess(lines)
3963enddef
3964
3965def Test_null_default_argument()
3966 var lines =<< trim END
3967 def Fstring(v: string = null_string)
3968 assert_equal(null_string, v)
3969 enddef
3970 Fstring()
3971 def Fblob(v: blob = null_blob)
3972 assert_equal(null_blob, v)
3973 enddef
3974 Fblob()
3975 def Flist(v: list<number> = null_list)
3976 assert_equal(null_list, v)
3977 enddef
3978 Flist()
3979 def Fdict(v: dict<number> = null_dict)
3980 assert_equal(null_dict, v)
3981 enddef
3982 Fdict()
3983 def Ffunc(Fv: func(number): number = null_function)
3984 assert_equal(null_function, Fv)
3985 enddef
3986 Ffunc()
3987 if has('channel')
3988 def Fchannel(v: channel = null_channel)
3989 assert_equal(null_channel, v)
3990 enddef
3991 Fchannel()
3992 def Fjob(v: job = null_job)
3993 assert_equal(null_job, v)
3994 enddef
3995 Fjob()
3996 endif
3997 END
3998 v9.CheckDefAndScriptSuccess(lines)
3999enddef
4000
4001def Test_null_return()
4002 var lines =<< trim END
4003 def Fstring(): string
4004 return null_string
4005 enddef
4006 assert_equal(null_string, Fstring())
4007 def Fblob(): blob
4008 return null_blob
4009 enddef
4010 assert_equal(null_blob, Fblob())
4011 def Flist(): list<number>
4012 return null_list
4013 enddef
4014 assert_equal(null_list, Flist())
4015 def Fdict(): dict<number>
4016 return null_dict
4017 enddef
4018 assert_equal(null_dict, Fdict())
4019 def Ffunc(): func(number): number
4020 return null_function
4021 enddef
4022 assert_equal(null_function, Ffunc())
4023 if has('channel')
4024 def Fchannel(): channel
4025 return null_channel
4026 enddef
4027 assert_equal(null_channel, Fchannel())
4028 def Fjob(): job
4029 return null_job
4030 enddef
4031 assert_equal(null_job, Fjob())
4032 endif
4033 END
4034 v9.CheckDefAndScriptSuccess(lines)
4035enddef
4036
Bram Moolenaar6e48b842021-08-10 22:52:02 +02004037def Test_list_any_type_checked()
4038 var lines =<< trim END
4039 vim9script
4040 def Foo()
4041 --decl--
4042 Bar(l)
4043 enddef
4044 def Bar(ll: list<dict<any>>)
4045 enddef
4046 Foo()
4047 END
Bram Moolenaar2d3ac2e2022-02-03 12:34:05 +00004048 # "any" could be "dict<any>", thus OK
Bram Moolenaar6e48b842021-08-10 22:52:02 +02004049 lines[2] = 'var l: list<any>'
Bram Moolenaar2d3ac2e2022-02-03 12:34:05 +00004050 v9.CheckScriptSuccess(lines)
Bram Moolenaar6e48b842021-08-10 22:52:02 +02004051 lines[2] = 'var l: list<any> = []'
Bram Moolenaar2d3ac2e2022-02-03 12:34:05 +00004052 v9.CheckScriptSuccess(lines)
Bram Moolenaar6e48b842021-08-10 22:52:02 +02004053
4054 lines[2] = 'var l: list<any> = [11]'
Bram Moolenaar62aec932022-01-29 21:45:34 +00004055 v9.CheckScriptFailure(lines, 'E1013: Argument 1: type mismatch, expected list<dict<any>> but got list<number>', 2)
Bram Moolenaar6e48b842021-08-10 22:52:02 +02004056enddef
4057
Bram Moolenaar701cc6c2021-04-10 13:33:48 +02004058def Test_compile_error()
4059 var lines =<< trim END
4060 def g:Broken()
4061 echo 'a' + {}
4062 enddef
4063 call g:Broken()
4064 END
4065 # First call: compilation error
Bram Moolenaar62aec932022-01-29 21:45:34 +00004066 v9.CheckScriptFailure(lines, 'E1051: Wrong argument type for +')
Bram Moolenaar701cc6c2021-04-10 13:33:48 +02004067
4068 # Second call won't try compiling again
4069 assert_fails('call g:Broken()', 'E1091: Function is not compiled: Broken')
Bram Moolenaar599410c2021-04-10 14:03:43 +02004070 delfunc g:Broken
4071
4072 # No error when compiling with :silent!
4073 lines =<< trim END
4074 def g:Broken()
4075 echo 'a' + []
4076 enddef
4077 silent! defcompile
4078 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00004079 v9.CheckScriptSuccess(lines)
Bram Moolenaar599410c2021-04-10 14:03:43 +02004080
4081 # Calling the function won't try compiling again
4082 assert_fails('call g:Broken()', 'E1091: Function is not compiled: Broken')
4083 delfunc g:Broken
Bram Moolenaar701cc6c2021-04-10 13:33:48 +02004084enddef
4085
Bram Moolenaar962c43b2021-04-10 17:18:09 +02004086def Test_ignored_argument()
4087 var lines =<< trim END
4088 vim9script
4089 def Ignore(_, _): string
4090 return 'yes'
4091 enddef
4092 assert_equal('yes', Ignore(1, 2))
4093
4094 func Ok(_)
4095 return a:_
4096 endfunc
4097 assert_equal('ok', Ok('ok'))
4098
4099 func Oktoo()
4100 let _ = 'too'
4101 return _
4102 endfunc
4103 assert_equal('too', Oktoo())
Bram Moolenaarda479c72021-04-10 21:01:38 +02004104
4105 assert_equal([[1], [2], [3]], range(3)->mapnew((_, v) => [v]->map((_, w) => w + 1)))
Bram Moolenaar962c43b2021-04-10 17:18:09 +02004106 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00004107 v9.CheckScriptSuccess(lines)
Bram Moolenaar962c43b2021-04-10 17:18:09 +02004108
4109 lines =<< trim END
4110 def Ignore(_: string): string
4111 return _
4112 enddef
4113 defcompile
4114 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00004115 v9.CheckScriptFailure(lines, 'E1181:', 1)
Bram Moolenaar962c43b2021-04-10 17:18:09 +02004116
4117 lines =<< trim END
4118 var _ = 1
4119 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00004120 v9.CheckDefAndScriptFailure(lines, 'E1181:', 1)
Yegappan Lakshmanan34fcb692021-05-25 20:14:00 +02004121
4122 lines =<< trim END
4123 var x = _
4124 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00004125 v9.CheckDefAndScriptFailure(lines, 'E1181:', 1)
Bram Moolenaar962c43b2021-04-10 17:18:09 +02004126enddef
4127
Bram Moolenaarbb8a7ce2021-04-10 20:10:26 +02004128def Test_too_many_arguments()
4129 var lines =<< trim END
4130 echo [0, 1, 2]->map(() => 123)
4131 END
Bram Moolenaareddd4fc2022-02-20 15:52:28 +00004132 v9.CheckDefAndScriptFailure(lines, ['E176:', 'E1106: 2 arguments too many'], 1)
Bram Moolenaarbb8a7ce2021-04-10 20:10:26 +02004133
4134 lines =<< trim END
4135 echo [0, 1, 2]->map((_) => 123)
4136 END
Bram Moolenaareddd4fc2022-02-20 15:52:28 +00004137 v9.CheckDefAndScriptFailure(lines, ['E176', 'E1106: One argument too many'], 1)
Bram Moolenaar31d99482022-05-26 22:24:43 +01004138
4139 lines =<< trim END
4140 vim9script
4141 def OneArgument(arg: string)
4142 echo arg
4143 enddef
4144 var Ref = OneArgument
4145 Ref('a', 'b')
4146 END
4147 v9.CheckScriptFailure(lines, 'E118:')
4148enddef
4149
4150def Test_funcref_with_base()
4151 var lines =<< trim END
4152 vim9script
4153 def TwoArguments(str: string, nr: number)
4154 echo str nr
4155 enddef
4156 var Ref = TwoArguments
4157 Ref('a', 12)
4158 'b'->Ref(34)
4159 END
4160 v9.CheckScriptSuccess(lines)
4161
4162 lines =<< trim END
4163 vim9script
4164 def TwoArguments(str: string, nr: number)
4165 echo str nr
4166 enddef
4167 var Ref = TwoArguments
4168 'a'->Ref('b')
4169 END
4170 v9.CheckScriptFailure(lines, 'E1013: Argument 2: type mismatch, expected number but got string', 6)
4171
4172 lines =<< trim END
4173 vim9script
4174 def TwoArguments(str: string, nr: number)
4175 echo str nr
4176 enddef
4177 var Ref = TwoArguments
4178 123->Ref(456)
4179 END
4180 v9.CheckScriptFailure(lines, 'E1013: Argument 1: type mismatch, expected string but got number')
4181
4182 lines =<< trim END
4183 vim9script
4184 def TwoArguments(nr: number, str: string)
4185 echo str nr
4186 enddef
4187 var Ref = TwoArguments
4188 123->Ref('b')
4189 def AndNowCompiled()
4190 456->Ref('x')
4191 enddef
4192 AndNowCompiled()
4193 END
4194 v9.CheckScriptSuccess(lines)
Bram Moolenaarbb8a7ce2021-04-10 20:10:26 +02004195enddef
Bram Moolenaar077a4232020-12-22 18:33:27 +01004196
Bram Moolenaara6aa1642021-04-23 19:32:23 +02004197def Test_closing_brace_at_start_of_line()
4198 var lines =<< trim END
4199 def Func()
4200 enddef
4201 Func(
4202 )
4203 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00004204 v9.CheckDefAndScriptSuccess(lines)
Bram Moolenaara6aa1642021-04-23 19:32:23 +02004205enddef
4206
Bram Moolenaar62aec932022-01-29 21:45:34 +00004207func s:CreateMydict()
Bram Moolenaarb033ee22021-08-15 16:08:36 +02004208 let g:mydict = {}
4209 func g:mydict.afunc()
4210 let g:result = self.key
4211 endfunc
4212endfunc
4213
4214def Test_numbered_function_reference()
4215 CreateMydict()
4216 var output = execute('legacy func g:mydict.afunc')
4217 var funcName = 'g:' .. substitute(output, '.*function \(\d\+\).*', '\1', '')
4218 execute 'function(' .. funcName .. ', [], {key: 42})()'
4219 # check that the function still exists
4220 assert_equal(output, execute('legacy func g:mydict.afunc'))
4221 unlet g:mydict
4222enddef
4223
Bram Moolenaarcfb4d4f2022-09-30 19:19:04 +01004224def Test_numbered_function_call()
4225 var lines =<< trim END
4226 let s:legacyscript = {}
4227 func s:legacyscript.Helper() abort
4228 return "Success"
4229 endfunc
4230 let g:legacyscript = deepcopy(s:legacyscript)
4231
4232 let g:legacy_result = eval("g:legacyscript.Helper()")
4233 vim9cmd g:vim9_result = eval("g:legacyscript.Helper()")
4234 END
4235 v9.CheckScriptSuccess(lines)
4236 assert_equal('Success', g:legacy_result)
4237 assert_equal('Success', g:vim9_result)
4238
4239 unlet g:legacy_result
4240 unlet g:vim9_result
4241enddef
4242
Bram Moolenaard3a11782022-01-05 16:50:40 +00004243def Test_go_beyond_end_of_cmd()
4244 # this was reading the byte after the end of the line
4245 var lines =<< trim END
4246 def F()
4247 cal
4248 enddef
4249 defcompile
4250 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00004251 v9.CheckScriptFailure(lines, 'E476:')
Bram Moolenaard3a11782022-01-05 16:50:40 +00004252enddef
4253
Yegappan Lakshmanan7c7e19c2022-04-09 11:09:07 +01004254" Test for memory allocation failure when defining a new lambda
4255func Test_lambda_allocation_failure()
4256 new
4257 let lines =<< trim END
4258 vim9script
4259 g:Xlambda = (x): number => {
4260 return x + 1
4261 }
4262 END
4263 call setline(1, lines)
4264 call test_alloc_fail(GetAllocId('get_func'), 0, 0)
4265 call assert_fails('source', 'E342:')
4266 call assert_false(exists('g:Xlambda'))
4267 bw!
4268endfunc
4269
Bram Moolenaarbce69d62022-05-22 13:45:52 +01004270def Test_multiple_funcref()
4271 # This was using a NULL pointer
4272 var lines =<< trim END
4273 vim9script
4274 def A(F: func, ...args: list<any>): func
4275 return funcref(F, args)
4276 enddef
4277
4278 def B(F: func): func
4279 return funcref(A, [F])
4280 enddef
4281
4282 def Test(n: number)
4283 enddef
4284
4285 const X = B(Test)
4286 X(1)
4287 END
4288 v9.CheckScriptSuccess(lines)
4289
4290 # slightly different case
4291 lines =<< trim END
4292 vim9script
4293
4294 def A(F: func, ...args: list<any>): any
4295 return call(F, args)
4296 enddef
4297
4298 def B(F: func): func
4299 return funcref(A, [F])
4300 enddef
4301
4302 def Test(n: number)
4303 enddef
4304
4305 const X = B(Test)
4306 X(1)
4307 END
4308 v9.CheckScriptSuccess(lines)
4309enddef
4310
Bram Moolenaarbd683e32022-07-18 17:49:03 +01004311def Test_cexpr_errmsg_line_number()
4312 var lines =<< trim END
4313 vim9script
4314 def Func()
4315 var qfl = {}
4316 cexpr qfl
4317 enddef
4318 Func()
4319 END
4320 v9.CheckScriptFailure(lines, 'E777', 2)
4321enddef
4322
Bram Moolenaar1d84f762022-09-03 21:35:53 +01004323def AddDefer(s: string)
4324 g:deferred->extend([s])
4325enddef
4326
4327def DeferTwo()
4328 g:deferred->extend(['in Two'])
4329 for n in range(3)
4330 defer g:AddDefer('two' .. n)
4331 endfor
4332 g:deferred->extend(['end Two'])
4333enddef
4334
4335def DeferOne()
4336 g:deferred->extend(['in One'])
4337 defer g:AddDefer('one')
4338 g:DeferTwo()
4339 g:deferred->extend(['end One'])
4340
4341 writefile(['text'], 'XdeferFile')
4342 defer delete('XdeferFile')
4343enddef
4344
4345def Test_defer()
4346 g:deferred = []
4347 g:DeferOne()
4348 assert_equal(['in One', 'in Two', 'end Two', 'two2', 'two1', 'two0', 'end One', 'one'], g:deferred)
4349 unlet g:deferred
4350 assert_equal('', glob('XdeferFile'))
4351enddef
4352
Bram Moolenaar3558afe2022-10-13 16:12:57 +01004353def Test_invalid_redir()
4354 var lines =<< trim END
4355 def Tone()
4356 if 1
4357 redi =>@�0
4358 redi END
4359 endif
4360 enddef
4361 defcompile
4362 END
4363 v9.CheckScriptFailure(lines, 'E354:')
4364 delfunc g:Tone
4365
4366 # this was reading past the end of the line
4367 lines =<< trim END
4368 def Ttwo()
4369 if 0
4370 redi =>@�0
4371 redi END
4372 endif
4373 enddef
4374 defcompile
4375 END
4376 v9.CheckScriptFailure(lines, 'E354:')
4377 delfunc g:Ttwo
4378enddef
4379
Bram Moolenaar8b716f52022-02-15 21:17:56 +00004380" The following messes up syntax highlight, keep near the end.
Bram Moolenaar20677332021-06-06 17:02:53 +02004381if has('python3')
Bram Moolenaar8b716f52022-02-15 21:17:56 +00004382 def Test_python3_command()
4383 py3 import vim
Bram Moolenaarf5288c52022-02-15 21:33:29 +00004384 py3 vim.command("g:done = 'yes'")
Bram Moolenaar8b716f52022-02-15 21:17:56 +00004385 assert_equal('yes', g:done)
4386 unlet g:done
4387 enddef
4388
Bram Moolenaar20677332021-06-06 17:02:53 +02004389 def Test_python3_heredoc()
4390 py3 << trim EOF
4391 import vim
4392 vim.vars['didit'] = 'yes'
4393 EOF
4394 assert_equal('yes', g:didit)
4395
4396 python3 << trim EOF
4397 import vim
4398 vim.vars['didit'] = 'again'
4399 EOF
4400 assert_equal('again', g:didit)
4401 enddef
4402endif
4403
Bram Moolenaar20677332021-06-06 17:02:53 +02004404if has('lua')
4405 def Test_lua_heredoc()
4406 g:d = {}
4407 lua << trim EOF
4408 x = vim.eval('g:d')
4409 x['key'] = 'val'
4410 EOF
4411 assert_equal('val', g:d.key)
4412 enddef
Bram Moolenaarefd73ae2022-03-20 18:51:00 +00004413
4414 def Test_lua_heredoc_fails()
4415 var lines = [
4416 'vim9script',
4417 'def ExeLua()',
4418 'lua << trim EOLUA',
4419 "x = vim.eval('g:nodict')",
4420 'EOLUA',
4421 'enddef',
4422 'ExeLua()',
4423 ]
4424 v9.CheckScriptFailure(lines, 'E121: Undefined variable: g:nodict')
4425 enddef
Bram Moolenaar20677332021-06-06 17:02:53 +02004426endif
4427
Bram Moolenaard881d152022-05-13 13:50:36 +01004428if has('perl')
4429 def Test_perl_heredoc_nested()
4430 var lines =<< trim END
4431 vim9script
4432 def F(): string
4433 def G(): string
4434 perl << EOF
4435 EOF
4436 return 'done'
4437 enddef
4438 return G()
4439 enddef
4440 assert_equal('done', F())
4441 END
4442 v9.CheckScriptSuccess(lines)
4443 enddef
4444endif
4445
Bram Moolenaarf7779c62020-05-03 15:38:16 +02004446
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02004447" vim: ts=8 sw=2 sts=2 expandtab tw=80 fdm=marker