blob: dc6b10439fd832c0ce5862815944217adcce91a8 [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
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200470let s:nothing = 0
471def ReturnNothing()
472 s:nothing = 1
473 if true
474 return
475 endif
476 s:nothing = 2
477enddef
478
479def Test_return_nothing()
Bram Moolenaar62aec932022-01-29 21:45:34 +0000480 g:ReturnNothing()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +0200481 s:nothing->assert_equal(1)
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200482enddef
483
Bram Moolenaar648ea762021-01-15 19:04:32 +0100484def Test_return_invalid()
485 var lines =<< trim END
486 vim9script
487 def Func(): invalid
488 return xxx
489 enddef
490 defcompile
491 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000492 v9.CheckScriptFailure(lines, 'E1010:', 2)
Bram Moolenaar31842cd2021-02-12 22:10:21 +0100493
494 lines =<< trim END
495 vim9script
496 def Test(Fun: func(number): number): list<number>
497 return map([1, 2, 3], (_, i) => Fun(i))
498 enddef
499 defcompile
500 def Inc(nr: number): nr
501 return nr + 2
502 enddef
503 echo Test(Inc)
504 END
505 # doing this twice was leaking memory
Bram Moolenaar62aec932022-01-29 21:45:34 +0000506 v9.CheckScriptFailure(lines, 'E1010:')
507 v9.CheckScriptFailure(lines, 'E1010:')
Bram Moolenaar648ea762021-01-15 19:04:32 +0100508enddef
509
Bram Moolenaarefc084e2021-09-09 22:30:52 +0200510def Test_return_list_any()
Bram Moolenaar114dbda2022-01-03 12:28:03 +0000511 # This used to fail but now the actual list type is checked, and since it has
512 # an item of type string it can be used as list<string>.
Bram Moolenaarefc084e2021-09-09 22:30:52 +0200513 var lines =<< trim END
514 vim9script
515 def Func(): list<string>
516 var l: list<any>
517 l->add('string')
518 return l
519 enddef
520 echo Func()
521 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000522 v9.CheckScriptSuccess(lines)
Bram Moolenaar114dbda2022-01-03 12:28:03 +0000523
Bram Moolenaarefc084e2021-09-09 22:30:52 +0200524 lines =<< trim END
525 vim9script
526 def Func(): list<string>
527 var l: list<any>
528 l += ['string']
529 return l
530 enddef
531 echo Func()
532 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000533 v9.CheckScriptSuccess(lines)
Bram Moolenaarefc084e2021-09-09 22:30:52 +0200534enddef
535
Bram Moolenaar1a572e92022-03-15 12:28:10 +0000536def Test_return_any_two_types()
537 var lines =<< trim END
538 vim9script
539
540 def G(Fn: func(string): any)
541 g:result = Fn("hello")
542 enddef
543
544 def F(a: number, b: string): any
545 echo b
546 if a > 0
547 return 1
548 else
549 return []
550 endif
551 enddef
552
553 G(function(F, [1]))
554 END
555 v9.CheckScriptSuccess(lines)
556 assert_equal(1, g:result)
557 unlet g:result
558enddef
559
Bram Moolenaar62aec932022-01-29 21:45:34 +0000560func s:Increment()
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200561 let g:counter += 1
562endfunc
563
564def Test_call_ufunc_count()
565 g:counter = 1
566 Increment()
567 Increment()
568 Increment()
Bram Moolenaarf5be8cd2020-07-17 20:36:00 +0200569 # works with and without :call
Bram Moolenaarc0c71e92020-09-11 19:09:48 +0200570 g:counter->assert_equal(4)
571 eval g:counter->assert_equal(4)
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200572 unlet g:counter
573enddef
574
Bram Moolenaar1983f1a2022-02-28 20:55:02 +0000575def Test_call_ufunc_failure()
576 var lines =<< trim END
577 vim9script
578 def Tryit()
579 g:Global(1, 2, 3)
580 enddef
581
582 func g:Global(a, b, c)
583 echo a:a a:b a:c
584 endfunc
585
586 defcompile
587
588 func! g:Global(a, b)
589 echo a:a a:b
590 endfunc
591 Tryit()
592 END
593 v9.CheckScriptFailure(lines, 'E118: Too many arguments for function: Global')
594 delfunc g:Global
595
596 lines =<< trim END
597 vim9script
598
599 g:Ref = function('len')
600 def Tryit()
601 g:Ref('x')
602 enddef
603
604 defcompile
605
606 g:Ref = function('add')
607 Tryit()
608 END
609 v9.CheckScriptFailure(lines, 'E119: Not enough arguments for function: add')
610 unlet g:Ref
611enddef
612
Bram Moolenaar62aec932022-01-29 21:45:34 +0000613def s:MyVarargs(arg: string, ...rest: list<string>): string
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +0200614 var res = arg
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200615 for s in rest
616 res ..= ',' .. s
617 endfor
618 return res
619enddef
620
621def Test_call_varargs()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +0200622 MyVarargs('one')->assert_equal('one')
623 MyVarargs('one', 'two')->assert_equal('one,two')
624 MyVarargs('one', 'two', 'three')->assert_equal('one,two,three')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200625enddef
626
Bram Moolenaar01dd6c32021-09-05 16:36:23 +0200627def Test_call_white_space()
Bram Moolenaar62aec932022-01-29 21:45:34 +0000628 v9.CheckDefAndScriptFailure(["call Test ('text')"], ['E476:', 'E1068:'])
Bram Moolenaar01dd6c32021-09-05 16:36:23 +0200629enddef
630
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200631def MyDefaultArgs(name = 'string'): string
632 return name
633enddef
634
Bram Moolenaar62aec932022-01-29 21:45:34 +0000635def s:MyDefaultSecond(name: string, second: bool = true): string
Bram Moolenaare30f64b2020-07-15 19:48:20 +0200636 return second ? name : 'none'
637enddef
638
Bram Moolenaar38a3bfa2021-03-29 22:14:55 +0200639
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200640def Test_call_default_args()
Bram Moolenaar62aec932022-01-29 21:45:34 +0000641 g:MyDefaultArgs()->assert_equal('string')
642 g:MyDefaultArgs(v:none)->assert_equal('string')
643 g:MyDefaultArgs('one')->assert_equal('one')
644 assert_fails('g:MyDefaultArgs("one", "two")', 'E118:', '', 4, 'Test_call_default_args')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200645
Bram Moolenaarc0c71e92020-09-11 19:09:48 +0200646 MyDefaultSecond('test')->assert_equal('test')
647 MyDefaultSecond('test', true)->assert_equal('test')
648 MyDefaultSecond('test', false)->assert_equal('none')
Bram Moolenaare30f64b2020-07-15 19:48:20 +0200649
Bram Moolenaar38a3bfa2021-03-29 22:14:55 +0200650 var lines =<< trim END
651 def MyDefaultThird(name: string, aa = 'aa', bb = 'bb'): string
652 return name .. aa .. bb
653 enddef
654
655 MyDefaultThird('->')->assert_equal('->aabb')
656 MyDefaultThird('->', v:none)->assert_equal('->aabb')
657 MyDefaultThird('->', 'xx')->assert_equal('->xxbb')
658 MyDefaultThird('->', v:none, v:none)->assert_equal('->aabb')
659 MyDefaultThird('->', 'xx', v:none)->assert_equal('->xxbb')
660 MyDefaultThird('->', v:none, 'yy')->assert_equal('->aayy')
661 MyDefaultThird('->', 'xx', 'yy')->assert_equal('->xxyy')
Bram Moolenaare28d9b32021-07-03 18:56:53 +0200662
663 def DefArg(mandatory: any, optional = mandatory): string
664 return mandatory .. optional
665 enddef
666 DefArg(1234)->assert_equal('12341234')
667 DefArg("ok")->assert_equal('okok')
Bram Moolenaar38a3bfa2021-03-29 22:14:55 +0200668 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000669 v9.CheckDefAndScriptSuccess(lines)
Bram Moolenaar38a3bfa2021-03-29 22:14:55 +0200670
Bram Moolenaar62aec932022-01-29 21:45:34 +0000671 v9.CheckScriptFailure(['def Func(arg: number = asdf)', 'enddef', 'defcompile'], 'E1001:')
Bram Moolenaar2d870f82020-12-05 13:41:01 +0100672 delfunc g:Func
Bram Moolenaar62aec932022-01-29 21:45:34 +0000673 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 +0100674 delfunc g:Func
Bram Moolenaar62aec932022-01-29 21:45:34 +0000675 v9.CheckDefFailure(['def Func(x: number = )', 'enddef'], 'E15:')
Bram Moolenaar12bce952021-03-11 20:04:04 +0100676
Bram Moolenaar38a3bfa2021-03-29 22:14:55 +0200677 lines =<< trim END
Bram Moolenaar12bce952021-03-11 20:04:04 +0100678 vim9script
679 def Func(a = b == 0 ? 1 : 2, b = 0)
680 enddef
681 defcompile
682 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000683 v9.CheckScriptFailure(lines, 'E1001: Variable not found: b')
Bram Moolenaar59618fe2021-12-21 12:32:17 +0000684
Bram Moolenaarfa46ead2021-12-22 13:18:39 +0000685 # using script variable requires matching type or type cast when executed
Bram Moolenaar59618fe2021-12-21 12:32:17 +0000686 lines =<< trim END
687 vim9script
688 var a: any
689 def Func(arg: string = a)
690 echo arg
691 enddef
692 defcompile
693 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000694 v9.CheckScriptSuccess(lines + ['a = "text"', 'Func()'])
695 v9.CheckScriptFailure(lines + ['a = 123', 'Func()'], 'E1013: Argument 1: type mismatch, expected string but got number')
Bram Moolenaar59618fe2021-12-21 12:32:17 +0000696
697 # using global variable does not require type cast
698 lines =<< trim END
699 vim9script
700 def Func(arg: string = g:str)
701 echo arg
702 enddef
703 g:str = 'works'
704 Func()
705 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000706 v9.CheckScriptSuccess(lines)
Bram Moolenaar04b12692020-05-04 23:24:44 +0200707enddef
708
Bram Moolenaar62aec932022-01-29 21:45:34 +0000709def s:FuncWithComment( # comment
Bram Moolenaarcef12702021-01-04 14:09:43 +0100710 a: number, #comment
711 b: bool, # comment
712 c: string) #comment
713 assert_equal(4, a)
714 assert_equal(true, b)
715 assert_equal('yes', c)
716enddef
717
718def Test_func_with_comments()
719 FuncWithComment(4, true, 'yes')
720
721 var lines =<< trim END
722 def Func(# comment
723 arg: string)
724 enddef
725 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000726 v9.CheckScriptFailure(lines, 'E125:', 1)
Bram Moolenaarcef12702021-01-04 14:09:43 +0100727
728 lines =<< trim END
729 def Func(
730 arg: string# comment
731 )
732 enddef
733 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000734 v9.CheckScriptFailure(lines, 'E475:', 2)
Bram Moolenaarcef12702021-01-04 14:09:43 +0100735
736 lines =<< trim END
737 def Func(
738 arg: string
739 )# comment
740 enddef
741 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000742 v9.CheckScriptFailure(lines, 'E488:', 3)
Bram Moolenaarcef12702021-01-04 14:09:43 +0100743enddef
744
Bram Moolenaar04b12692020-05-04 23:24:44 +0200745def Test_nested_function()
Bram Moolenaar38453522021-11-28 22:00:12 +0000746 def NestedDef(arg: string): string
Bram Moolenaar04b12692020-05-04 23:24:44 +0200747 return 'nested ' .. arg
748 enddef
Bram Moolenaar38453522021-11-28 22:00:12 +0000749 NestedDef(':def')->assert_equal('nested :def')
750
751 func NestedFunc(arg)
752 return 'nested ' .. a:arg
753 endfunc
754 NestedFunc(':func')->assert_equal('nested :func')
Bram Moolenaar04b12692020-05-04 23:24:44 +0200755
Bram Moolenaar62aec932022-01-29 21:45:34 +0000756 v9.CheckDefFailure(['def Nested()', 'enddef', 'Nested(66)'], 'E118:')
757 v9.CheckDefFailure(['def Nested(arg: string)', 'enddef', 'Nested()'], 'E119:')
Bram Moolenaar0e65d3d2020-05-05 17:53:16 +0200758
Bram Moolenaar62aec932022-01-29 21:45:34 +0000759 v9.CheckDefFailure(['def s:Nested()', 'enddef'], 'E1075:')
760 v9.CheckDefFailure(['def b:Nested()', 'enddef'], 'E1075:')
Bram Moolenaar8b848ca2020-09-10 22:28:01 +0200761
Bram Moolenaar54021752020-12-06 18:50:36 +0100762 var lines =<< trim END
763 def Outer()
764 def Inner()
765 # comment
766 enddef
767 def Inner()
768 enddef
769 enddef
770 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000771 v9.CheckDefFailure(lines, 'E1073:')
Bram Moolenaar54021752020-12-06 18:50:36 +0100772
773 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, 'E1117:')
Bram Moolenaar54021752020-12-06 18:50:36 +0100783
Bram Moolenaardb8e5c22021-12-25 19:58:22 +0000784 lines =<< trim END
785 vim9script
786 def Outer()
787 def Inner()
788 g:result = 'ok'
789 enddef
790 Inner()
791 enddef
792 Outer()
793 Inner()
794 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000795 v9.CheckScriptFailure(lines, 'E117: Unknown function: Inner')
Bram Moolenaardb8e5c22021-12-25 19:58:22 +0000796 assert_equal('ok', g:result)
797 unlet g:result
798
Bram Moolenaarf681cfb2022-02-07 20:30:57 +0000799 lines =<< trim END
800 vim9script
801 def Outer()
802 def _Inner()
803 echo 'bad'
804 enddef
Bram Moolenaar3787f262022-02-07 21:54:01 +0000805 _Inner()
Bram Moolenaarf681cfb2022-02-07 20:30:57 +0000806 enddef
807 defcompile
808 END
Bram Moolenaar3787f262022-02-07 21:54:01 +0000809 v9.CheckScriptFailure(lines, 'E1267:')
Bram Moolenaarf681cfb2022-02-07 20:30:57 +0000810
811 lines =<< trim END
812 vim9script
813 def Outer()
814 def g:inner()
815 echo 'bad'
816 enddef
Bram Moolenaar3787f262022-02-07 21:54:01 +0000817 g:inner()
Bram Moolenaarf681cfb2022-02-07 20:30:57 +0000818 enddef
819 defcompile
820 END
Bram Moolenaar3787f262022-02-07 21:54:01 +0000821 v9.CheckScriptFailure(lines, 'E1267:')
822
823 lines =<< trim END
824 vim9script
825 def g:_Func()
826 echo 'bad'
827 enddef
828 END
829 v9.CheckScriptFailure(lines, 'E1267:')
830
831 lines =<< trim END
832 vim9script
Bram Moolenaara749a422022-02-12 19:52:25 +0000833 def _Func()
Bram Moolenaar3787f262022-02-07 21:54:01 +0000834 echo 'bad'
835 enddef
836 END
837 v9.CheckScriptFailure(lines, 'E1267:')
Bram Moolenaarf681cfb2022-02-07 20:30:57 +0000838
Bram Moolenaar54021752020-12-06 18:50:36 +0100839 # nested function inside conditional
Bram Moolenaar54021752020-12-06 18:50:36 +0100840 lines =<< trim END
841 vim9script
842 var thecount = 0
843 if true
844 def Test(): number
845 def TheFunc(): number
846 thecount += 1
847 return thecount
848 enddef
849 return TheFunc()
850 enddef
851 endif
852 defcompile
853 assert_equal(1, Test())
854 assert_equal(2, Test())
855 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000856 v9.CheckScriptSuccess(lines)
Bram Moolenaar8863bda2021-03-17 18:42:08 +0100857
858 # also works when "thecount" is inside the "if" block
859 lines =<< trim END
860 vim9script
861 if true
862 var thecount = 0
863 def Test(): number
864 def TheFunc(): number
865 thecount += 1
866 return thecount
867 enddef
868 return TheFunc()
869 enddef
870 endif
871 defcompile
872 assert_equal(1, Test())
873 assert_equal(2, Test())
874 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000875 v9.CheckScriptSuccess(lines)
Bram Moolenaar4bba16d2021-08-15 19:28:05 +0200876
Bram Moolenaara915fa02022-03-23 11:29:15 +0000877 # nested function with recursive call
878 lines =<< trim END
879 vim9script
880
881 def MyFunc(): number
882 def Fib(n: number): number
883 if n < 2
884 return 1
885 endif
886 return Fib(n - 2) + Fib(n - 1)
887 enddef
888
889 return Fib(5)
890 enddef
891
892 assert_equal(8, MyFunc())
893 END
894 v9.CheckScriptSuccess(lines)
895
Bram Moolenaar4bba16d2021-08-15 19:28:05 +0200896 lines =<< trim END
897 vim9script
898 def Outer()
899 def Inner()
900 echo 'hello'
901 enddef burp
902 enddef
903 defcompile
904 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000905 v9.CheckScriptFailure(lines, 'E1173: Text found after enddef: burp', 3)
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200906enddef
907
Bram Moolenaar1889f492022-08-16 19:34:44 +0100908def Test_nested_function_fails()
909 var lines =<< trim END
910 def T()
911 def Func(g: string):string
912 enddef
913 Func()
914 enddef
915 silent! defcompile
916 END
917 v9.CheckScriptFailure(lines, 'E1069:')
918enddef
919
Bram Moolenaaradc8e442020-12-31 18:28:18 +0100920def Test_not_nested_function()
921 echo printf('%d',
922 function('len')('xxx'))
923enddef
924
Bram Moolenaaraf8edbb2020-08-01 00:03:09 +0200925func Test_call_default_args_from_func()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +0200926 call MyDefaultArgs()->assert_equal('string')
927 call MyDefaultArgs('one')->assert_equal('one')
Bram Moolenaar9bd5d872020-09-06 21:47:48 +0200928 call assert_fails('call MyDefaultArgs("one", "two")', 'E118:', '', 3, 'Test_call_default_args_from_func')
Bram Moolenaaraf8edbb2020-08-01 00:03:09 +0200929endfunc
930
Bram Moolenaar38ddf332020-07-31 22:05:04 +0200931def Test_nested_global_function()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +0200932 var lines =<< trim END
Bram Moolenaar38ddf332020-07-31 22:05:04 +0200933 vim9script
934 def Outer()
935 def g:Inner(): string
936 return 'inner'
937 enddef
938 enddef
Bram Moolenaaraf8edbb2020-08-01 00:03:09 +0200939 defcompile
940 Outer()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +0200941 g:Inner()->assert_equal('inner')
Bram Moolenaaraf8edbb2020-08-01 00:03:09 +0200942 delfunc g:Inner
943 Outer()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +0200944 g:Inner()->assert_equal('inner')
Bram Moolenaaraf8edbb2020-08-01 00:03:09 +0200945 delfunc g:Inner
946 Outer()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +0200947 g:Inner()->assert_equal('inner')
Bram Moolenaaraf8edbb2020-08-01 00:03:09 +0200948 delfunc g:Inner
Bram Moolenaar38ddf332020-07-31 22:05:04 +0200949 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000950 v9.CheckScriptSuccess(lines)
Bram Moolenaar2c79e9d2020-08-01 18:57:52 +0200951
952 lines =<< trim END
953 vim9script
954 def Outer()
Bram Moolenaar38453522021-11-28 22:00:12 +0000955 func g:Inner()
956 return 'inner'
957 endfunc
958 enddef
959 defcompile
960 Outer()
961 g:Inner()->assert_equal('inner')
962 delfunc g:Inner
963 Outer()
964 g:Inner()->assert_equal('inner')
965 delfunc g:Inner
966 Outer()
967 g:Inner()->assert_equal('inner')
968 delfunc g:Inner
969 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000970 v9.CheckScriptSuccess(lines)
Bram Moolenaar38453522021-11-28 22:00:12 +0000971
972 lines =<< trim END
973 vim9script
974 def Outer()
Bram Moolenaar2c79e9d2020-08-01 18:57:52 +0200975 def g:Inner(): string
976 return 'inner'
977 enddef
978 enddef
979 defcompile
980 Outer()
981 Outer()
982 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000983 v9.CheckScriptFailure(lines, "E122:")
Bram Moolenaarcd45ed02020-12-22 17:35:54 +0100984 delfunc g:Inner
Bram Moolenaarad486a02020-08-01 23:22:18 +0200985
986 lines =<< trim END
987 vim9script
Bram Moolenaar58a52f22020-12-22 18:56:55 +0100988 def Outer()
989 def g:Inner()
Bram Moolenaar2949cfd2020-12-31 21:28:47 +0100990 echo map([1, 2, 3], (_, v) => v + 1)
Bram Moolenaar58a52f22020-12-22 18:56:55 +0100991 enddef
992 g:Inner()
993 enddef
994 Outer()
995 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000996 v9.CheckScriptSuccess(lines)
Bram Moolenaar58a52f22020-12-22 18:56:55 +0100997 delfunc g:Inner
998
999 lines =<< trim END
1000 vim9script
Bram Moolenaarad486a02020-08-01 23:22:18 +02001001 def Func()
1002 echo 'script'
1003 enddef
1004 def Outer()
1005 def Func()
1006 echo 'inner'
1007 enddef
1008 enddef
1009 defcompile
1010 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001011 v9.CheckScriptFailure(lines, "E1073:", 1)
Bram Moolenaard604d782021-11-20 21:46:20 +00001012
1013 lines =<< trim END
1014 vim9script
1015 def Func()
1016 echo 'script'
1017 enddef
1018 def Func()
1019 echo 'script'
1020 enddef
1021 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001022 v9.CheckScriptFailure(lines, "E1073:", 5)
Bram Moolenaar38ddf332020-07-31 22:05:04 +02001023enddef
1024
Bram Moolenaar6abdcf82020-11-22 18:15:44 +01001025def DefListAll()
1026 def
1027enddef
1028
1029def DefListOne()
1030 def DefListOne
1031enddef
1032
1033def DefListMatches()
1034 def /DefList
1035enddef
1036
1037def Test_nested_def_list()
1038 var funcs = split(execute('call DefListAll()'), "\n")
1039 assert_true(len(funcs) > 10)
1040 assert_true(funcs->index('def DefListAll()') >= 0)
1041
1042 funcs = split(execute('call DefListOne()'), "\n")
1043 assert_equal([' def DefListOne()', '1 def DefListOne', ' enddef'], funcs)
1044
1045 funcs = split(execute('call DefListMatches()'), "\n")
1046 assert_true(len(funcs) >= 3)
1047 assert_true(funcs->index('def DefListAll()') >= 0)
1048 assert_true(funcs->index('def DefListOne()') >= 0)
1049 assert_true(funcs->index('def DefListMatches()') >= 0)
Bram Moolenaar54021752020-12-06 18:50:36 +01001050
1051 var lines =<< trim END
1052 vim9script
1053 def Func()
1054 def +Func+
1055 enddef
1056 defcompile
1057 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001058 v9.CheckScriptFailure(lines, 'E476:', 1)
Bram Moolenaar6abdcf82020-11-22 18:15:44 +01001059enddef
1060
Bram Moolenaare08be092022-02-17 13:08:26 +00001061def Test_global_function_not_found()
1062 var lines =<< trim END
1063 g:Ref = 123
1064 call g:Ref()
1065 END
1066 v9.CheckDefExecAndScriptFailure(lines, ['E117:', 'E1085:'], 2)
1067enddef
1068
Bram Moolenaar333894b2020-08-01 18:53:07 +02001069def Test_global_local_function()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001070 var lines =<< trim END
Bram Moolenaar333894b2020-08-01 18:53:07 +02001071 vim9script
1072 def g:Func(): string
1073 return 'global'
1074 enddef
1075 def Func(): string
1076 return 'local'
1077 enddef
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001078 g:Func()->assert_equal('global')
1079 Func()->assert_equal('local')
Bram Moolenaar2d870f82020-12-05 13:41:01 +01001080 delfunc g:Func
Bram Moolenaar333894b2020-08-01 18:53:07 +02001081 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001082 v9.CheckScriptSuccess(lines)
Bram Moolenaar035d6e92020-08-11 22:30:42 +02001083
1084 lines =<< trim END
1085 vim9script
1086 def g:Funcy()
1087 echo 'funcy'
1088 enddef
Bram Moolenaara749a422022-02-12 19:52:25 +00001089 Funcy()
Bram Moolenaar035d6e92020-08-11 22:30:42 +02001090 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001091 v9.CheckScriptFailure(lines, 'E117:')
Bram Moolenaar333894b2020-08-01 18:53:07 +02001092enddef
1093
Bram Moolenaar0f769812020-09-12 18:32:34 +02001094def Test_local_function_shadows_global()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001095 var lines =<< trim END
Bram Moolenaar0f769812020-09-12 18:32:34 +02001096 vim9script
1097 def g:Gfunc(): string
1098 return 'global'
1099 enddef
1100 def AnotherFunc(): number
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001101 var Gfunc = function('len')
Bram Moolenaar0f769812020-09-12 18:32:34 +02001102 return Gfunc('testing')
1103 enddef
1104 g:Gfunc()->assert_equal('global')
1105 AnotherFunc()->assert_equal(7)
1106 delfunc g:Gfunc
1107 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001108 v9.CheckScriptSuccess(lines)
Bram Moolenaar0f769812020-09-12 18:32:34 +02001109
1110 lines =<< trim END
1111 vim9script
1112 def g:Func(): string
1113 return 'global'
1114 enddef
1115 def AnotherFunc()
1116 g:Func = function('len')
1117 enddef
1118 AnotherFunc()
1119 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001120 v9.CheckScriptFailure(lines, 'E705:')
Bram Moolenaar0f769812020-09-12 18:32:34 +02001121 delfunc g:Func
Bram Moolenaar0865b152021-04-05 15:38:51 +02001122
Bram Moolenaar62aec932022-01-29 21:45:34 +00001123 # global function is not found with g: prefix
Bram Moolenaar0865b152021-04-05 15:38:51 +02001124 lines =<< trim END
1125 vim9script
1126 def g:Func(): string
1127 return 'global'
1128 enddef
1129 def AnotherFunc(): string
1130 return Func()
1131 enddef
1132 assert_equal('global', AnotherFunc())
Bram Moolenaar0865b152021-04-05 15:38:51 +02001133 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001134 v9.CheckScriptFailure(lines, 'E117:')
1135 delfunc g:Func
Bram Moolenaar0865b152021-04-05 15:38:51 +02001136
1137 lines =<< trim END
1138 vim9script
1139 def g:Func(): string
1140 return 'global'
1141 enddef
Bram Moolenaar848fadd2022-01-30 15:28:30 +00001142 assert_equal('global', g:Func())
Bram Moolenaar0865b152021-04-05 15:38:51 +02001143 delfunc g:Func
1144 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001145 v9.CheckScriptSuccess(lines)
Bram Moolenaar58493cf2022-01-06 12:23:30 +00001146
1147 # This does not shadow "i" which is visible only inside the for loop
1148 lines =<< trim END
1149 vim9script
1150
1151 def Foo(i: number)
1152 echo i
1153 enddef
1154
1155 for i in range(3)
1156 # Foo() is compiled here
1157 Foo(i)
1158 endfor
1159 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001160 v9.CheckScriptSuccess(lines)
Bram Moolenaar0f769812020-09-12 18:32:34 +02001161enddef
1162
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001163func TakesOneArg(arg)
1164 echo a:arg
1165endfunc
1166
1167def Test_call_wrong_args()
Bram Moolenaar62aec932022-01-29 21:45:34 +00001168 v9.CheckDefFailure(['g:TakesOneArg()'], 'E119:')
1169 v9.CheckDefFailure(['g:TakesOneArg(11, 22)'], 'E118:')
1170 v9.CheckDefFailure(['bufnr(xxx)'], 'E1001:')
1171 v9.CheckScriptFailure(['def Func(Ref: func(s: string))'], 'E475:')
Bram Moolenaaree8580e2020-08-28 17:19:07 +02001172
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001173 var lines =<< trim END
Bram Moolenaaree8580e2020-08-28 17:19:07 +02001174 vim9script
1175 def Func(s: string)
1176 echo s
1177 enddef
1178 Func([])
1179 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001180 v9.CheckScriptFailure(lines, 'E1013: Argument 1: type mismatch, expected string but got list<unknown>', 5)
Bram Moolenaarb185a402020-09-18 22:42:00 +02001181
Bram Moolenaar9a015112021-12-31 14:06:45 +00001182 # argument name declared earlier is found when declaring a function
Bram Moolenaarb185a402020-09-18 22:42:00 +02001183 lines =<< trim END
1184 vim9script
Bram Moolenaarb4893b82021-02-21 22:20:24 +01001185 var name = 'piet'
1186 def FuncOne(name: string)
Bram Moolenaar3a5988c2022-02-08 19:23:35 +00001187 echo name
Bram Moolenaarb4893b82021-02-21 22:20:24 +01001188 enddef
1189 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001190 v9.CheckScriptFailure(lines, 'E1168:')
Bram Moolenaarb4893b82021-02-21 22:20:24 +01001191
Bram Moolenaar3a5988c2022-02-08 19:23:35 +00001192 # same, inside the same block
1193 lines =<< trim END
1194 vim9script
1195 if true
1196 var name = 'piet'
1197 def FuncOne(name: string)
1198 echo name
1199 enddef
1200 endif
1201 END
1202 v9.CheckScriptFailure(lines, 'E1168:')
1203
1204 # variable in other block is OK
1205 lines =<< trim END
1206 vim9script
1207 if true
1208 var name = 'piet'
1209 endif
1210 def FuncOne(name: string)
1211 echo name
1212 enddef
1213 END
1214 v9.CheckScriptSuccess(lines)
1215
Bram Moolenaardce24412022-02-08 20:35:30 +00001216 # with another variable in another block
1217 lines =<< trim END
1218 vim9script
1219 if true
1220 var name = 'piet'
1221 # define a function so that the variable isn't cleared
1222 def GetItem(): string
1223 return item
1224 enddef
1225 endif
1226 if true
1227 var name = 'peter'
1228 def FuncOne(name: string)
1229 echo name
1230 enddef
1231 endif
1232 END
1233 v9.CheckScriptFailure(lines, 'E1168:')
1234
1235 # only variable in another block is OK
1236 lines =<< trim END
1237 vim9script
1238 if true
1239 var name = 'piet'
1240 # define a function so that the variable isn't cleared
1241 def GetItem(): string
1242 return item
1243 enddef
1244 endif
1245 if true
1246 def FuncOne(name: string)
1247 echo name
1248 enddef
1249 endif
1250 END
1251 v9.CheckScriptSuccess(lines)
1252
Bram Moolenaar9a015112021-12-31 14:06:45 +00001253 # argument name declared later is only found when compiling
1254 lines =<< trim END
1255 vim9script
1256 def FuncOne(name: string)
1257 echo nr
1258 enddef
1259 var name = 'piet'
1260 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001261 v9.CheckScriptSuccess(lines)
1262 v9.CheckScriptFailure(lines + ['defcompile'], 'E1168:')
Bram Moolenaar9a015112021-12-31 14:06:45 +00001263
Bram Moolenaarb4893b82021-02-21 22:20:24 +01001264 lines =<< trim END
1265 vim9script
Bram Moolenaarb185a402020-09-18 22:42:00 +02001266 def FuncOne(nr: number)
1267 echo nr
1268 enddef
1269 def FuncTwo()
1270 FuncOne()
1271 enddef
1272 defcompile
1273 END
1274 writefile(lines, 'Xscript')
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001275 var didCatch = false
Bram Moolenaarb185a402020-09-18 22:42:00 +02001276 try
1277 source Xscript
1278 catch
1279 assert_match('E119: Not enough arguments for function: <SNR>\d\+_FuncOne', v:exception)
1280 assert_match('Xscript\[8\]..function <SNR>\d\+_FuncTwo, line 1', v:throwpoint)
1281 didCatch = true
1282 endtry
1283 assert_true(didCatch)
1284
1285 lines =<< trim END
1286 vim9script
1287 def FuncOne(nr: number)
1288 echo nr
1289 enddef
1290 def FuncTwo()
1291 FuncOne(1, 2)
1292 enddef
1293 defcompile
1294 END
Bram Moolenaarf5fec052022-09-11 11:49:22 +01001295 writefile(lines, 'Xscript', 'D')
Bram Moolenaarb185a402020-09-18 22:42:00 +02001296 didCatch = false
1297 try
1298 source Xscript
1299 catch
1300 assert_match('E118: Too many arguments for function: <SNR>\d\+_FuncOne', v:exception)
1301 assert_match('Xscript\[8\]..function <SNR>\d\+_FuncTwo, line 1', v:throwpoint)
1302 didCatch = true
1303 endtry
1304 assert_true(didCatch)
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001305enddef
1306
Bram Moolenaar50824712020-12-20 21:10:17 +01001307def Test_call_funcref_wrong_args()
1308 var head =<< trim END
1309 vim9script
1310 def Func3(a1: string, a2: number, a3: list<number>)
1311 echo a1 .. a2 .. a3[0]
1312 enddef
1313 def Testme()
1314 var funcMap: dict<func> = {func: Func3}
1315 END
1316 var tail =<< trim END
1317 enddef
1318 Testme()
1319 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001320 v9.CheckScriptSuccess(head + ["funcMap['func']('str', 123, [1, 2, 3])"] + tail)
Bram Moolenaar50824712020-12-20 21:10:17 +01001321
Bram Moolenaar62aec932022-01-29 21:45:34 +00001322 v9.CheckScriptFailure(head + ["funcMap['func']('str', 123)"] + tail, 'E119:')
1323 v9.CheckScriptFailure(head + ["funcMap['func']('str', 123, [1], 4)"] + tail, 'E118:')
Bram Moolenaar32b3f822021-01-06 21:59:39 +01001324
1325 var lines =<< trim END
1326 vim9script
1327 var Ref: func(number): any
1328 Ref = (j) => !j
1329 echo Ref(false)
1330 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001331 v9.CheckScriptFailure(lines, 'E1013: Argument 1: type mismatch, expected number but got bool', 4)
Bram Moolenaar32b3f822021-01-06 21:59:39 +01001332
1333 lines =<< trim END
1334 vim9script
1335 var Ref: func(number): any
1336 Ref = (j) => !j
1337 call Ref(false)
1338 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001339 v9.CheckScriptFailure(lines, 'E1013: Argument 1: type mismatch, expected number but got bool', 4)
Bram Moolenaar50824712020-12-20 21:10:17 +01001340enddef
1341
Bram Moolenaarb4d16cb2020-11-05 18:45:46 +01001342def Test_call_lambda_args()
Bram Moolenaar2a389082021-04-09 20:24:31 +02001343 var lines =<< trim END
1344 var Callback = (..._) => 'anything'
1345 assert_equal('anything', Callback())
1346 assert_equal('anything', Callback(1))
1347 assert_equal('anything', Callback('a', 2))
Bram Moolenaar1088b692021-04-09 22:12:44 +02001348
1349 assert_equal('xyz', ((a: string): string => a)('xyz'))
Bram Moolenaar2a389082021-04-09 20:24:31 +02001350 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001351 v9.CheckDefAndScriptSuccess(lines)
Bram Moolenaar2a389082021-04-09 20:24:31 +02001352
Bram Moolenaar62aec932022-01-29 21:45:34 +00001353 v9.CheckDefFailure(['echo ((i) => 0)()'],
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01001354 'E119: Not enough arguments for function: ((i) => 0)()')
Bram Moolenaarb4d16cb2020-11-05 18:45:46 +01001355
Bram Moolenaar2a389082021-04-09 20:24:31 +02001356 lines =<< trim END
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01001357 var Ref = (x: number, y: number) => x + y
Bram Moolenaarb4d16cb2020-11-05 18:45:46 +01001358 echo Ref(1, 'x')
1359 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001360 v9.CheckDefFailure(lines, 'E1013: Argument 2: type mismatch, expected number but got string')
Bram Moolenaare68b02a2021-01-03 13:09:51 +01001361
1362 lines =<< trim END
1363 var Ref: func(job, string, number)
1364 Ref = (x, y) => 0
1365 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001366 v9.CheckDefAndScriptFailure(lines, 'E1012:')
Bram Moolenaare68b02a2021-01-03 13:09:51 +01001367
1368 lines =<< trim END
1369 var Ref: func(job, string)
1370 Ref = (x, y, z) => 0
1371 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001372 v9.CheckDefAndScriptFailure(lines, 'E1012:')
Bram Moolenaar057e84a2021-02-28 16:55:11 +01001373
1374 lines =<< trim END
1375 var one = 1
1376 var l = [1, 2, 3]
1377 echo map(l, (one) => one)
1378 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001379 v9.CheckDefFailure(lines, 'E1167:')
1380 v9.CheckScriptFailure(['vim9script'] + lines, 'E1168:')
Bram Moolenaar057e84a2021-02-28 16:55:11 +01001381
1382 lines =<< trim END
Bram Moolenaar14ded112021-06-26 19:25:49 +02001383 var Ref: func(any, ?any): bool
1384 Ref = (_, y = 1) => false
1385 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001386 v9.CheckDefAndScriptFailure(lines, 'E1172:')
Bram Moolenaar14ded112021-06-26 19:25:49 +02001387
1388 lines =<< trim END
Bram Moolenaar015cf102021-06-26 21:52:02 +02001389 var a = 0
1390 var b = (a == 0 ? 1 : 2)
1391 assert_equal(1, b)
Bram Moolenaar98f9a5f2021-06-26 22:22:38 +02001392 var txt = 'a'
1393 b = (txt =~ 'x' ? 1 : 2)
1394 assert_equal(2, b)
Bram Moolenaar015cf102021-06-26 21:52:02 +02001395 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001396 v9.CheckDefAndScriptSuccess(lines)
Bram Moolenaar015cf102021-06-26 21:52:02 +02001397
1398 lines =<< trim END
Bram Moolenaar057e84a2021-02-28 16:55:11 +01001399 def ShadowLocal()
1400 var one = 1
1401 var l = [1, 2, 3]
1402 echo map(l, (one) => one)
1403 enddef
1404 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001405 v9.CheckDefFailure(lines, 'E1167:')
Bram Moolenaar057e84a2021-02-28 16:55:11 +01001406
1407 lines =<< trim END
1408 def Shadowarg(one: number)
1409 var l = [1, 2, 3]
1410 echo map(l, (one) => one)
1411 enddef
1412 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001413 v9.CheckDefFailure(lines, 'E1167:')
Bram Moolenaar767034c2021-04-09 17:24:52 +02001414
1415 lines =<< trim END
1416 echo ((a) => a)('aa', 'bb')
1417 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001418 v9.CheckDefAndScriptFailure(lines, 'E118:', 1)
Bram Moolenaarc4c56422021-07-21 20:38:46 +02001419
1420 lines =<< trim END
1421 echo 'aa'->((a) => a)('bb')
1422 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001423 v9.CheckDefFailure(lines, 'E118: Too many arguments for function: ->((a) => a)(''bb'')', 1)
1424 v9.CheckScriptFailure(['vim9script'] + lines, 'E118: Too many arguments for function: <lambda>', 2)
Bram Moolenaarb4d16cb2020-11-05 18:45:46 +01001425enddef
1426
Bram Moolenaara755fdb2021-11-20 21:35:41 +00001427def Test_lambda_line_nr()
1428 var lines =<< trim END
1429 vim9script
1430 # comment
1431 # comment
1432 var id = timer_start(1'000, (_) => 0)
1433 var out = execute('verbose ' .. timer_info(id)[0].callback
1434 ->string()
1435 ->substitute("('\\|')", ' ', 'g'))
1436 assert_match('Last set from .* line 4', out)
1437 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001438 v9.CheckScriptSuccess(lines)
Bram Moolenaara755fdb2021-11-20 21:35:41 +00001439enddef
1440
Bram Moolenaar5f91e742021-03-17 21:29:29 +01001441def FilterWithCond(x: string, Cond: func(string): bool): bool
1442 return Cond(x)
1443enddef
1444
Bram Moolenaar0346b792021-01-31 22:18:29 +01001445def Test_lambda_return_type()
1446 var lines =<< trim END
1447 var Ref = (): => 123
1448 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001449 v9.CheckDefAndScriptFailure(lines, 'E1157:', 1)
Bram Moolenaar5f91e742021-03-17 21:29:29 +01001450
Yegappan Lakshmanan611728f2021-05-24 15:15:47 +02001451 # no space before the return type
1452 lines =<< trim END
1453 var Ref = (x):number => x + 1
1454 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001455 v9.CheckDefAndScriptFailure(lines, 'E1069:', 1)
Yegappan Lakshmanan611728f2021-05-24 15:15:47 +02001456
Bram Moolenaar5f91e742021-03-17 21:29:29 +01001457 # this works
1458 for x in ['foo', 'boo']
Bram Moolenaar62aec932022-01-29 21:45:34 +00001459 echo g:FilterWithCond(x, (v) => v =~ '^b')
Bram Moolenaar5f91e742021-03-17 21:29:29 +01001460 endfor
1461
1462 # this fails
1463 lines =<< trim END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001464 echo g:FilterWithCond('foo', (v) => v .. '^b')
Bram Moolenaar5f91e742021-03-17 21:29:29 +01001465 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001466 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 +02001467
1468 lines =<< trim END
1469 var Lambda1 = (x) => {
1470 return x
1471 }
1472 assert_equal('asdf', Lambda1('asdf'))
1473 var Lambda2 = (x): string => {
1474 return x
1475 }
1476 assert_equal('foo', Lambda2('foo'))
1477 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001478 v9.CheckDefAndScriptSuccess(lines)
Bram Moolenaara9931532021-06-12 15:58:16 +02001479
1480 lines =<< trim END
1481 var Lambda = (x): string => {
1482 return x
1483 }
1484 echo Lambda(['foo'])
1485 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001486 v9.CheckDefExecAndScriptFailure(lines, 'E1012:')
Bram Moolenaar0346b792021-01-31 22:18:29 +01001487enddef
1488
Bram Moolenaar709664c2020-12-12 14:33:41 +01001489def Test_lambda_uses_assigned_var()
Bram Moolenaar62aec932022-01-29 21:45:34 +00001490 v9.CheckDefSuccess([
Bram Moolenaar2984ed32022-08-20 14:51:17 +01001491 'var x: any = "aaa"',
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01001492 'x = filter(["bbb"], (_, v) => v =~ x)'])
Bram Moolenaar709664c2020-12-12 14:33:41 +01001493enddef
1494
Bram Moolenaarf5f4e852022-09-22 22:03:14 +01001495def Test_lambda_invalid_block()
1496 var lines =<< trim END
1497 timer_start(0, (_) => { # echo
1498 echo 'yes'
1499 })
1500 END
1501 v9.CheckDefAndScriptSuccess(lines)
1502
1503 lines =<< trim END
1504 timer_start(0, (_) => { " echo
1505 echo 'yes'
1506 })
1507 END
1508 v9.CheckDefAndScriptFailure(lines, 'E488: Trailing characters: " echo')
1509
1510 lines =<< trim END
1511 timer_start(0, (_) => { | echo
1512 echo 'yes'
1513 })
1514 END
1515 v9.CheckDefAndScriptFailure(lines, 'E488: Trailing characters: | echo')
1516enddef
1517
Bram Moolenaarf8addf12022-09-23 12:44:25 +01001518def Test_lambda_with_following_cmd()
1519 var lines =<< trim END
1520 set ts=2
1521 var Lambda = () => {
1522 set ts=4
1523 } | set ts=3
1524 assert_equal(3, &ts)
1525 Lambda()
1526 assert_equal(4, &ts)
1527 END
1528 v9.CheckDefAndScriptSuccess(lines)
1529 set ts=8
1530enddef
1531
Bram Moolenaar18062fc2021-03-05 21:35:47 +01001532def Test_pass_legacy_lambda_to_def_func()
1533 var lines =<< trim END
1534 vim9script
1535 func Foo()
1536 eval s:Bar({x -> 0})
1537 endfunc
1538 def Bar(y: any)
1539 enddef
1540 Foo()
1541 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001542 v9.CheckScriptSuccess(lines)
Bram Moolenaar831bdf82021-06-22 19:32:17 +02001543
1544 lines =<< trim END
1545 vim9script
Bram Moolenaar1d9cef72022-03-17 16:30:03 +00001546 def g:TestFunc(F: func)
Bram Moolenaar831bdf82021-06-22 19:32:17 +02001547 enddef
1548 legacy call g:TestFunc({-> 0})
1549 delfunc g:TestFunc
1550
Bram Moolenaar1d9cef72022-03-17 16:30:03 +00001551 def g:TestFunc(F: func(number))
Bram Moolenaar831bdf82021-06-22 19:32:17 +02001552 enddef
1553 legacy call g:TestFunc({nr -> 0})
1554 delfunc g:TestFunc
1555 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001556 v9.CheckScriptSuccess(lines)
Bram Moolenaar18062fc2021-03-05 21:35:47 +01001557enddef
1558
Bram Moolenaar844fb642021-10-23 13:32:30 +01001559def Test_lambda_in_reduce_line_break()
1560 # this was using freed memory
1561 var lines =<< trim END
1562 vim9script
1563 const result: dict<number> =
1564 ['Bob', 'Sam', 'Cat', 'Bob', 'Cat', 'Cat']
1565 ->reduce((acc, val) => {
1566 if has_key(acc, val)
1567 acc[val] += 1
1568 return acc
1569 else
1570 acc[val] = 1
1571 return acc
1572 endif
1573 }, {})
1574 assert_equal({Bob: 2, Sam: 1, Cat: 3}, result)
1575 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001576 v9.CheckScriptSuccess(lines)
Bram Moolenaar844fb642021-10-23 13:32:30 +01001577enddef
1578
Bram Moolenaardcb53be2021-12-09 14:23:43 +00001579def Test_set_opfunc_to_lambda()
1580 var lines =<< trim END
1581 vim9script
1582 nnoremap <expr> <F4> <SID>CountSpaces() .. '_'
1583 def CountSpaces(type = ''): string
1584 if type == ''
1585 &operatorfunc = (t) => CountSpaces(t)
1586 return 'g@'
1587 endif
1588 normal! '[V']y
1589 g:result = getreg('"')->count(' ')
1590 return ''
1591 enddef
1592 new
1593 'a b c d e'->setline(1)
1594 feedkeys("\<F4>", 'x')
1595 assert_equal(4, g:result)
1596 bwipe!
1597 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001598 v9.CheckScriptSuccess(lines)
Bram Moolenaardcb53be2021-12-09 14:23:43 +00001599enddef
1600
Bram Moolenaaref082e12021-12-12 21:02:03 +00001601def Test_set_opfunc_to_global_function()
1602 var lines =<< trim END
1603 vim9script
1604 def g:CountSpaces(type = ''): string
1605 normal! '[V']y
1606 g:result = getreg('"')->count(' ')
1607 return ''
1608 enddef
Bram Moolenaarb15cf442021-12-16 15:49:43 +00001609 # global function works at script level
Bram Moolenaaref082e12021-12-12 21:02:03 +00001610 &operatorfunc = g:CountSpaces
1611 new
1612 'a b c d e'->setline(1)
1613 feedkeys("g@_", 'x')
1614 assert_equal(4, g:result)
Bram Moolenaarb15cf442021-12-16 15:49:43 +00001615
1616 &operatorfunc = ''
1617 g:result = 0
1618 # global function works in :def function
1619 def Func()
1620 &operatorfunc = g:CountSpaces
1621 enddef
1622 Func()
1623 feedkeys("g@_", 'x')
1624 assert_equal(4, g:result)
1625
Bram Moolenaaref082e12021-12-12 21:02:03 +00001626 bwipe!
1627 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001628 v9.CheckScriptSuccess(lines)
Bram Moolenaaref082e12021-12-12 21:02:03 +00001629 &operatorfunc = ''
1630enddef
1631
Bram Moolenaar33b968d2021-12-13 11:31:04 +00001632def Test_use_script_func_name_with_prefix()
1633 var lines =<< trim END
1634 vim9script
Bram Moolenaara749a422022-02-12 19:52:25 +00001635 func g:Getit()
Bram Moolenaar33b968d2021-12-13 11:31:04 +00001636 return 'it'
1637 endfunc
Bram Moolenaara749a422022-02-12 19:52:25 +00001638 var Fn = g:Getit
Bram Moolenaar33b968d2021-12-13 11:31:04 +00001639 assert_equal('it', Fn())
1640 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001641 v9.CheckScriptSuccess(lines)
Bram Moolenaar33b968d2021-12-13 11:31:04 +00001642enddef
1643
Bram Moolenaardd297bc2021-12-10 10:37:38 +00001644def Test_lambda_type_allocated()
1645 # Check that unreferencing a partial using a lambda can use the variable type
1646 # after the lambda has been freed and does not leak memory.
1647 var lines =<< trim END
1648 vim9script
1649
1650 func MyomniFunc1(val, findstart, base)
1651 return a:findstart ? 0 : []
1652 endfunc
1653
1654 var Lambda = (a, b) => MyomniFunc1(19, a, b)
1655 &omnifunc = Lambda
1656 Lambda = (a, b) => MyomniFunc1(20, a, b)
1657 &omnifunc = string(Lambda)
1658 Lambda = (a, b) => strlen(a)
1659 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001660 v9.CheckScriptSuccess(lines)
Bram Moolenaardd297bc2021-12-10 10:37:38 +00001661enddef
1662
Bram Moolenaara7583c42022-05-07 21:14:05 +01001663def Test_define_lambda_in_execute()
1664 var lines =<< trim [CODE]
1665 vim9script
1666
1667 def BuildFuncMultiLine(): func
1668 var x =<< trim END
1669 g:SomeRandomFunc = (d: dict<any>) => {
1670 return d.k1 + d.k2
1671 }
1672 END
1673 execute(x)
1674 return g:SomeRandomFunc
1675 enddef
1676 var ResultPlus = BuildFuncMultiLine()
1677 assert_equal(7, ResultPlus({k1: 3, k2: 4}))
1678 [CODE]
1679 v9.CheckScriptSuccess(lines)
1680 unlet g:SomeRandomFunc
1681enddef
1682
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001683" Default arg and varargs
1684def MyDefVarargs(one: string, two = 'foo', ...rest: list<string>): string
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001685 var res = one .. ',' .. two
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001686 for s in rest
1687 res ..= ',' .. s
1688 endfor
1689 return res
1690enddef
1691
1692def Test_call_def_varargs()
Bram Moolenaar62aec932022-01-29 21:45:34 +00001693 assert_fails('g:MyDefVarargs()', 'E119:', '', 1, 'Test_call_def_varargs')
1694 g:MyDefVarargs('one')->assert_equal('one,foo')
1695 g:MyDefVarargs('one', 'two')->assert_equal('one,two')
1696 g:MyDefVarargs('one', 'two', 'three')->assert_equal('one,two,three')
1697 v9.CheckDefFailure(['g:MyDefVarargs("one", 22)'],
Bram Moolenaar77072282020-09-16 17:55:40 +02001698 'E1013: Argument 2: type mismatch, expected string but got number')
Bram Moolenaar62aec932022-01-29 21:45:34 +00001699 v9.CheckDefFailure(['g:MyDefVarargs("one", "two", 123)'],
Bram Moolenaar77072282020-09-16 17:55:40 +02001700 'E1013: Argument 3: type mismatch, expected string but got number')
Bram Moolenaar24aa48b2020-07-25 16:33:02 +02001701
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001702 var lines =<< trim END
Bram Moolenaar24aa48b2020-07-25 16:33:02 +02001703 vim9script
1704 def Func(...l: list<string>)
1705 echo l
1706 enddef
1707 Func('a', 'b', 'c')
1708 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001709 v9.CheckScriptSuccess(lines)
Bram Moolenaar24aa48b2020-07-25 16:33:02 +02001710
1711 lines =<< trim END
1712 vim9script
1713 def Func(...l: list<string>)
1714 echo l
1715 enddef
1716 Func()
1717 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001718 v9.CheckScriptSuccess(lines)
Bram Moolenaar24aa48b2020-07-25 16:33:02 +02001719
1720 lines =<< trim END
1721 vim9script
Bram Moolenaar2a389082021-04-09 20:24:31 +02001722 def Func(...l: list<any>)
Bram Moolenaar2f8cbc42020-09-16 17:22:59 +02001723 echo l
1724 enddef
1725 Func(0)
1726 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001727 v9.CheckScriptSuccess(lines)
Bram Moolenaar2f8cbc42020-09-16 17:22:59 +02001728
1729 lines =<< trim END
1730 vim9script
Bram Moolenaar2a389082021-04-09 20:24:31 +02001731 def Func(...l: any)
1732 echo l
1733 enddef
1734 Func(0)
1735 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001736 v9.CheckScriptFailure(lines, 'E1180:', 2)
Bram Moolenaar2a389082021-04-09 20:24:31 +02001737
1738 lines =<< trim END
1739 vim9script
Bram Moolenaar28022722020-09-21 22:02:49 +02001740 def Func(..._l: list<string>)
1741 echo _l
1742 enddef
1743 Func('a', 'b', 'c')
1744 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001745 v9.CheckScriptSuccess(lines)
Bram Moolenaar28022722020-09-21 22:02:49 +02001746
1747 lines =<< trim END
1748 vim9script
Bram Moolenaar24aa48b2020-07-25 16:33:02 +02001749 def Func(...l: list<string>)
1750 echo l
1751 enddef
1752 Func(1, 2, 3)
1753 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001754 v9.CheckScriptFailure(lines, 'E1013: Argument 1: type mismatch')
Bram Moolenaar24aa48b2020-07-25 16:33:02 +02001755
1756 lines =<< trim END
1757 vim9script
1758 def Func(...l: list<string>)
1759 echo l
1760 enddef
1761 Func('a', 9)
1762 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001763 v9.CheckScriptFailure(lines, 'E1013: Argument 2: type mismatch')
Bram Moolenaar24aa48b2020-07-25 16:33:02 +02001764
1765 lines =<< trim END
1766 vim9script
1767 def Func(...l: list<string>)
1768 echo l
1769 enddef
1770 Func(1, 'a')
1771 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001772 v9.CheckScriptFailure(lines, 'E1013: Argument 1: type mismatch')
Bram Moolenaar4f53b792021-02-07 15:59:49 +01001773
1774 lines =<< trim END
1775 vim9script
1776 def Func( # some comment
1777 ...l = []
1778 )
1779 echo l
1780 enddef
1781 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001782 v9.CheckScriptFailure(lines, 'E1160:')
Bram Moolenaar6ce46b92021-08-07 15:35:36 +02001783
1784 lines =<< trim END
1785 vim9script
1786 def DoIt()
1787 g:Later('')
1788 enddef
1789 defcompile
1790 def g:Later(...l: list<number>)
1791 enddef
1792 DoIt()
1793 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001794 v9.CheckScriptFailure(lines, 'E1013: Argument 1: type mismatch, expected number but got string')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001795enddef
1796
Bram Moolenaar1378fbc2020-04-11 20:50:33 +02001797let s:value = ''
1798
1799def FuncOneDefArg(opt = 'text')
1800 s:value = opt
1801enddef
1802
1803def FuncTwoDefArg(nr = 123, opt = 'text'): string
1804 return nr .. opt
1805enddef
1806
1807def FuncVarargs(...arg: list<string>): string
1808 return join(arg, ',')
1809enddef
1810
1811def Test_func_type_varargs()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001812 var RefDefArg: func(?string)
Bram Moolenaar848fadd2022-01-30 15:28:30 +00001813 RefDefArg = g:FuncOneDefArg
Bram Moolenaar1378fbc2020-04-11 20:50:33 +02001814 RefDefArg()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001815 s:value->assert_equal('text')
Bram Moolenaar1378fbc2020-04-11 20:50:33 +02001816 RefDefArg('some')
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001817 s:value->assert_equal('some')
Bram Moolenaar1378fbc2020-04-11 20:50:33 +02001818
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001819 var RefDef2Arg: func(?number, ?string): string
Bram Moolenaar848fadd2022-01-30 15:28:30 +00001820 RefDef2Arg = g:FuncTwoDefArg
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001821 RefDef2Arg()->assert_equal('123text')
1822 RefDef2Arg(99)->assert_equal('99text')
1823 RefDef2Arg(77, 'some')->assert_equal('77some')
Bram Moolenaar1378fbc2020-04-11 20:50:33 +02001824
Bram Moolenaar62aec932022-01-29 21:45:34 +00001825 v9.CheckDefFailure(['var RefWrong: func(string?)'], 'E1010:')
1826 v9.CheckDefFailure(['var RefWrong: func(?string, string)'], 'E1007:')
Bram Moolenaar1378fbc2020-04-11 20:50:33 +02001827
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001828 var RefVarargs: func(...list<string>): string
Bram Moolenaar848fadd2022-01-30 15:28:30 +00001829 RefVarargs = g:FuncVarargs
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001830 RefVarargs()->assert_equal('')
1831 RefVarargs('one')->assert_equal('one')
1832 RefVarargs('one', 'two')->assert_equal('one,two')
Bram Moolenaar1378fbc2020-04-11 20:50:33 +02001833
Bram Moolenaar62aec932022-01-29 21:45:34 +00001834 v9.CheckDefFailure(['var RefWrong: func(...list<string>, string)'], 'E110:')
1835 v9.CheckDefFailure(['var RefWrong: func(...list<string>, ?string)'], 'E110:')
Bram Moolenaar1378fbc2020-04-11 20:50:33 +02001836enddef
1837
Bram Moolenaar0b76b422020-04-07 22:05:08 +02001838" Only varargs
1839def MyVarargsOnly(...args: list<string>): string
1840 return join(args, ',')
1841enddef
1842
1843def Test_call_varargs_only()
Bram Moolenaar62aec932022-01-29 21:45:34 +00001844 g:MyVarargsOnly()->assert_equal('')
1845 g:MyVarargsOnly('one')->assert_equal('one')
1846 g:MyVarargsOnly('one', 'two')->assert_equal('one,two')
1847 v9.CheckDefFailure(['g:MyVarargsOnly(1)'], 'E1013: Argument 1: type mismatch, expected string but got number')
1848 v9.CheckDefFailure(['g:MyVarargsOnly("one", 2)'], 'E1013: Argument 2: type mismatch, expected string but got number')
Bram Moolenaar0b76b422020-04-07 22:05:08 +02001849enddef
1850
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001851def Test_using_var_as_arg()
Bram Moolenaard2939812021-12-30 17:09:05 +00001852 var lines =<< trim END
1853 def Func(x: number)
1854 var x = 234
1855 enddef
1856 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001857 v9.CheckDefFailure(lines, 'E1006:')
Bram Moolenaard2939812021-12-30 17:09:05 +00001858
1859 lines =<< trim END
1860 def Func(Ref: number)
1861 def Ref()
1862 enddef
1863 enddef
1864 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001865 v9.CheckDefFailure(lines, 'E1073:')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001866enddef
1867
Bram Moolenaar62aec932022-01-29 21:45:34 +00001868def s:DictArg(arg: dict<string>)
Bram Moolenaarcb2bdb12020-05-10 22:53:56 +02001869 arg['key'] = 'value'
1870enddef
1871
Bram Moolenaar62aec932022-01-29 21:45:34 +00001872def s:ListArg(arg: list<string>)
Bram Moolenaarcb2bdb12020-05-10 22:53:56 +02001873 arg[0] = 'value'
1874enddef
1875
1876def Test_assign_to_argument()
Bram Moolenaarf5be8cd2020-07-17 20:36:00 +02001877 # works for dict and list
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001878 var d: dict<string> = {}
Bram Moolenaarcb2bdb12020-05-10 22:53:56 +02001879 DictArg(d)
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001880 d['key']->assert_equal('value')
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001881 var l: list<string> = []
Bram Moolenaarcb2bdb12020-05-10 22:53:56 +02001882 ListArg(l)
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001883 l[0]->assert_equal('value')
Bram Moolenaarcb2bdb12020-05-10 22:53:56 +02001884
Bram Moolenaar62aec932022-01-29 21:45:34 +00001885 v9.CheckScriptFailure(['def Func(arg: number)', 'arg = 3', 'enddef', 'defcompile'], 'E1090:')
Bram Moolenaar2d870f82020-12-05 13:41:01 +01001886 delfunc! g:Func
Bram Moolenaarcb2bdb12020-05-10 22:53:56 +02001887enddef
1888
Bram Moolenaarb816dae2020-09-20 22:04:00 +02001889" These argument names are reserved in legacy functions.
Bram Moolenaar62aec932022-01-29 21:45:34 +00001890def s:WithReservedNames(firstline: string, lastline: string): string
Bram Moolenaarb816dae2020-09-20 22:04:00 +02001891 return firstline .. lastline
1892enddef
1893
1894def Test_argument_names()
1895 assert_equal('OK', WithReservedNames('O', 'K'))
1896enddef
1897
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001898def Test_call_func_defined_later()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001899 g:DefinedLater('one')->assert_equal('one')
Bram Moolenaar9bd5d872020-09-06 21:47:48 +02001900 assert_fails('NotDefined("one")', 'E117:', '', 2, 'Test_call_func_defined_later')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001901enddef
1902
Bram Moolenaar1df8b3f2020-04-23 18:13:23 +02001903func DefinedLater(arg)
1904 return a:arg
1905endfunc
1906
1907def Test_call_funcref()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001908 g:SomeFunc('abc')->assert_equal(3)
Bram Moolenaar9bd5d872020-09-06 21:47:48 +02001909 assert_fails('NotAFunc()', 'E117:', '', 2, 'Test_call_funcref') # comment after call
Bram Moolenaar2ef91562021-12-11 16:14:07 +00001910 assert_fails('g:NotAFunc()', 'E1085:', '', 3, 'Test_call_funcref')
Bram Moolenaar2f1980f2020-07-22 19:30:06 +02001911
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001912 var lines =<< trim END
Bram Moolenaar2f1980f2020-07-22 19:30:06 +02001913 vim9script
1914 def RetNumber(): number
1915 return 123
1916 enddef
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001917 var Funcref: func: number = function('RetNumber')
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001918 Funcref()->assert_equal(123)
Bram Moolenaar2f1980f2020-07-22 19:30:06 +02001919 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001920 v9.CheckScriptSuccess(lines)
Bram Moolenaar0f60e802020-07-22 20:16:11 +02001921
1922 lines =<< trim END
1923 vim9script
1924 def RetNumber(): number
1925 return 123
1926 enddef
1927 def Bar(F: func: number): number
1928 return F()
1929 enddef
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001930 var Funcref = function('RetNumber')
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001931 Bar(Funcref)->assert_equal(123)
Bram Moolenaar0f60e802020-07-22 20:16:11 +02001932 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001933 v9.CheckScriptSuccess(lines)
Bram Moolenaarbfba8652020-07-23 20:09:10 +02001934
1935 lines =<< trim END
1936 vim9script
1937 def UseNumber(nr: number)
1938 echo nr
1939 enddef
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001940 var Funcref: func(number) = function('UseNumber')
Bram Moolenaarbfba8652020-07-23 20:09:10 +02001941 Funcref(123)
1942 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001943 v9.CheckScriptSuccess(lines)
Bram Moolenaarb8070e32020-07-23 20:56:04 +02001944
1945 lines =<< trim END
1946 vim9script
1947 def UseNumber(nr: number)
1948 echo nr
1949 enddef
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001950 var Funcref: func(string) = function('UseNumber')
Bram Moolenaarb8070e32020-07-23 20:56:04 +02001951 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001952 v9.CheckScriptFailure(lines, 'E1012: Type mismatch; expected func(string) but got func(number)')
Bram Moolenaar4fc224c2020-07-26 17:56:25 +02001953
1954 lines =<< trim END
1955 vim9script
1956 def EchoNr(nr = 34)
1957 g:echo = nr
1958 enddef
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001959 var Funcref: func(?number) = function('EchoNr')
Bram Moolenaar4fc224c2020-07-26 17:56:25 +02001960 Funcref()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001961 g:echo->assert_equal(34)
Bram Moolenaar4fc224c2020-07-26 17:56:25 +02001962 Funcref(123)
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001963 g:echo->assert_equal(123)
Bram Moolenaar4fc224c2020-07-26 17:56:25 +02001964 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001965 v9.CheckScriptSuccess(lines)
Bram Moolenaarace61322020-07-26 18:16:58 +02001966
1967 lines =<< trim END
1968 vim9script
1969 def EchoList(...l: list<number>)
1970 g:echo = l
1971 enddef
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001972 var Funcref: func(...list<number>) = function('EchoList')
Bram Moolenaarace61322020-07-26 18:16:58 +02001973 Funcref()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001974 g:echo->assert_equal([])
Bram Moolenaarace61322020-07-26 18:16:58 +02001975 Funcref(1, 2, 3)
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001976 g:echo->assert_equal([1, 2, 3])
Bram Moolenaarace61322020-07-26 18:16:58 +02001977 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001978 v9.CheckScriptSuccess(lines)
Bram Moolenaar01865ad2020-07-26 18:33:09 +02001979
1980 lines =<< trim END
1981 vim9script
1982 def OptAndVar(nr: number, opt = 12, ...l: list<number>): number
1983 g:optarg = opt
1984 g:listarg = l
1985 return nr
1986 enddef
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001987 var Funcref: func(number, ?number, ...list<number>): number = function('OptAndVar')
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001988 Funcref(10)->assert_equal(10)
1989 g:optarg->assert_equal(12)
1990 g:listarg->assert_equal([])
Bram Moolenaar01865ad2020-07-26 18:33:09 +02001991
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001992 Funcref(11, 22)->assert_equal(11)
1993 g:optarg->assert_equal(22)
1994 g:listarg->assert_equal([])
Bram Moolenaar01865ad2020-07-26 18:33:09 +02001995
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001996 Funcref(17, 18, 1, 2, 3)->assert_equal(17)
1997 g:optarg->assert_equal(18)
1998 g:listarg->assert_equal([1, 2, 3])
Bram Moolenaar01865ad2020-07-26 18:33:09 +02001999 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002000 v9.CheckScriptSuccess(lines)
Kota Kato948a3892022-08-16 16:09:59 +01002001
2002 lines =<< trim END
2003 function s:func(num)
2004 return a:num * 2
2005 endfunction
2006
2007 def s:CallFuncref()
2008 var Funcref = function('s:func')
2009 Funcref(3)->assert_equal(6)
2010 enddef
2011 call s:CallFuncref()
2012 END
2013 v9.CheckScriptSuccess(lines)
2014
2015 lines =<< trim END
2016 function s:func(num)
2017 return a:num * 2
2018 endfunction
2019
2020 def s:CallFuncref()
2021 var Funcref = function(s:func)
2022 Funcref(3)->assert_equal(6)
2023 enddef
2024 call s:CallFuncref()
2025 END
2026 v9.CheckScriptSuccess(lines)
2027
2028 lines =<< trim END
2029 function s:func(num)
2030 return a:num * 2
2031 endfunction
2032
2033 def s:CallFuncref()
2034 var Funcref = s:func
2035 Funcref(3)->assert_equal(6)
2036 enddef
2037 call s:CallFuncref()
2038 END
2039 v9.CheckScriptSuccess(lines)
Bram Moolenaar1df8b3f2020-04-23 18:13:23 +02002040enddef
2041
2042let SomeFunc = function('len')
2043let NotAFunc = 'text'
2044
Bram Moolenaar99aaf0c2020-04-12 14:39:53 +02002045def CombineFuncrefTypes()
Bram Moolenaarf5be8cd2020-07-17 20:36:00 +02002046 # same arguments, different return type
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002047 var Ref1: func(bool): string
2048 var Ref2: func(bool): number
2049 var Ref3: func(bool): any
Bram Moolenaar99aaf0c2020-04-12 14:39:53 +02002050 Ref3 = g:cond ? Ref1 : Ref2
2051
Bram Moolenaarf5be8cd2020-07-17 20:36:00 +02002052 # different number of arguments
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002053 var Refa1: func(bool): number
2054 var Refa2: func(bool, number): number
2055 var Refa3: func: number
Bram Moolenaar99aaf0c2020-04-12 14:39:53 +02002056 Refa3 = g:cond ? Refa1 : Refa2
2057
Bram Moolenaarf5be8cd2020-07-17 20:36:00 +02002058 # different argument types
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002059 var Refb1: func(bool, string): number
2060 var Refb2: func(string, number): number
2061 var Refb3: func(any, any): number
Bram Moolenaar99aaf0c2020-04-12 14:39:53 +02002062 Refb3 = g:cond ? Refb1 : Refb2
2063enddef
2064
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002065def FuncWithForwardCall()
Bram Moolenaar1df8b3f2020-04-23 18:13:23 +02002066 return g:DefinedEvenLater("yes")
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002067enddef
2068
2069def DefinedEvenLater(arg: string): string
2070 return arg
2071enddef
2072
2073def Test_error_in_nested_function()
Bram Moolenaarf5be8cd2020-07-17 20:36:00 +02002074 # Error in called function requires unwinding the call stack.
Bram Moolenaar848fadd2022-01-30 15:28:30 +00002075 assert_fails('g:FuncWithForwardCall()', 'E1096:', '', 1, 'FuncWithForwardCall')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002076enddef
2077
Bram Moolenaar4bf10062021-12-28 17:23:12 +00002078def Test_nested_function_with_nextcmd()
Bram Moolenaar9c23f9b2021-12-26 14:23:22 +00002079 var lines =<< trim END
2080 vim9script
2081 # Define an outer function
2082 def FirstFunction()
2083 # Define an inner function
2084 def SecondFunction()
2085 # the function has a body, a double free is detected.
2086 AAAAA
2087
2088 # enddef followed by | or } followed by # one or more characters
2089 enddef|BBBB
2090 enddef
2091
2092 # Compile all functions
2093 defcompile
2094 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002095 v9.CheckScriptFailure(lines, 'E1173: Text found after enddef: BBBB')
Bram Moolenaar9c23f9b2021-12-26 14:23:22 +00002096enddef
2097
Bram Moolenaar4bf10062021-12-28 17:23:12 +00002098def Test_nested_function_with_args_split()
2099 var lines =<< trim END
2100 vim9script
2101 def FirstFunction()
2102 def SecondFunction(
2103 )
2104 # had a double free if the right parenthesis of the nested function is
2105 # on the next line
2106
2107 enddef|BBBB
2108 enddef
2109 # Compile all functions
2110 defcompile
2111 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002112 v9.CheckScriptFailure(lines, 'E1173: Text found after enddef: BBBB')
Bram Moolenaar7473a842021-12-28 17:55:26 +00002113
2114 lines =<< trim END
2115 vim9script
2116 def FirstFunction()
2117 func SecondFunction()
2118 endfunc|BBBB
2119 enddef
2120 defcompile
2121 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002122 v9.CheckScriptFailure(lines, 'E1173: Text found after endfunction: BBBB')
Bram Moolenaar4bf10062021-12-28 17:23:12 +00002123enddef
2124
Bram Moolenaar9f1a39a2022-01-08 15:39:39 +00002125def Test_error_in_function_args()
2126 var lines =<< trim END
2127 def FirstFunction()
2128 def SecondFunction(J =
2129 # Nois
2130 # one
2131
2132 enddef|BBBB
2133 enddef
2134 # Compile all functions
2135 defcompile
2136 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002137 v9.CheckScriptFailure(lines, 'E488:')
Bram Moolenaar9f1a39a2022-01-08 15:39:39 +00002138enddef
2139
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002140def Test_return_type_wrong()
Bram Moolenaar62aec932022-01-29 21:45:34 +00002141 v9.CheckScriptFailure([
Bram Moolenaar5a849da2020-08-08 16:47:30 +02002142 'def Func(): number',
2143 'return "a"',
2144 'enddef',
2145 'defcompile'], 'expected number but got string')
Bram Moolenaar2d870f82020-12-05 13:41:01 +01002146 delfunc! g:Func
Bram Moolenaar62aec932022-01-29 21:45:34 +00002147 v9.CheckScriptFailure([
Bram Moolenaar5a849da2020-08-08 16:47:30 +02002148 'def Func(): string',
2149 'return 1',
2150 'enddef',
2151 'defcompile'], 'expected string but got number')
Bram Moolenaar2d870f82020-12-05 13:41:01 +01002152 delfunc! g:Func
Bram Moolenaar62aec932022-01-29 21:45:34 +00002153 v9.CheckScriptFailure([
Bram Moolenaar5a849da2020-08-08 16:47:30 +02002154 'def Func(): void',
2155 'return "a"',
2156 'enddef',
2157 'defcompile'],
2158 'E1096: Returning a value in a function without a return type')
Bram Moolenaar2d870f82020-12-05 13:41:01 +01002159 delfunc! g:Func
Bram Moolenaar62aec932022-01-29 21:45:34 +00002160 v9.CheckScriptFailure([
Bram Moolenaar5a849da2020-08-08 16:47:30 +02002161 'def Func()',
2162 'return "a"',
2163 'enddef',
2164 'defcompile'],
2165 'E1096: Returning a value in a function without a return type')
Bram Moolenaar2d870f82020-12-05 13:41:01 +01002166 delfunc! g:Func
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002167
Bram Moolenaar62aec932022-01-29 21:45:34 +00002168 v9.CheckScriptFailure([
Bram Moolenaar5a849da2020-08-08 16:47:30 +02002169 'def Func(): number',
2170 'return',
2171 'enddef',
2172 'defcompile'], 'E1003:')
Bram Moolenaar2d870f82020-12-05 13:41:01 +01002173 delfunc! g:Func
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002174
Bram Moolenaar62aec932022-01-29 21:45:34 +00002175 v9.CheckScriptFailure([
Bram Moolenaar33ea9fd2021-08-08 19:07:37 +02002176 'def Func():number',
2177 'return 123',
2178 'enddef',
2179 'defcompile'], 'E1069:')
2180 delfunc! g:Func
2181
Bram Moolenaar62aec932022-01-29 21:45:34 +00002182 v9.CheckScriptFailure([
Bram Moolenaar33ea9fd2021-08-08 19:07:37 +02002183 'def Func() :number',
2184 'return 123',
2185 'enddef',
2186 'defcompile'], 'E1059:')
2187 delfunc! g:Func
2188
Bram Moolenaar62aec932022-01-29 21:45:34 +00002189 v9.CheckScriptFailure([
Bram Moolenaar33ea9fd2021-08-08 19:07:37 +02002190 'def Func() : number',
2191 'return 123',
2192 'enddef',
2193 'defcompile'], 'E1059:')
2194 delfunc! g:Func
2195
Bram Moolenaar62e0e2e2022-08-20 12:07:58 +01002196 v9.CheckScriptFailure(['def Func(): list', 'return []', 'enddef'], 'E1008: Missing <type> after list')
Bram Moolenaar2d870f82020-12-05 13:41:01 +01002197 delfunc! g:Func
Bram Moolenaar62e0e2e2022-08-20 12:07:58 +01002198 v9.CheckScriptFailure(['def Func(): dict', 'return {}', 'enddef'], 'E1008: Missing <type> after dict')
Bram Moolenaar2d870f82020-12-05 13:41:01 +01002199 delfunc! g:Func
Bram Moolenaar62aec932022-01-29 21:45:34 +00002200 v9.CheckScriptFailure(['def Func()', 'return 1'], 'E1057:')
Bram Moolenaar2d870f82020-12-05 13:41:01 +01002201 delfunc! g:Func
Bram Moolenaar5a849da2020-08-08 16:47:30 +02002202
Bram Moolenaar62aec932022-01-29 21:45:34 +00002203 v9.CheckScriptFailure([
Bram Moolenaar5a849da2020-08-08 16:47:30 +02002204 'vim9script',
2205 'def FuncB()',
2206 ' return 123',
2207 'enddef',
2208 'def FuncA()',
2209 ' FuncB()',
2210 'enddef',
2211 'defcompile'], 'E1096:')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002212enddef
2213
2214def Test_arg_type_wrong()
Bram Moolenaar62e0e2e2022-08-20 12:07:58 +01002215 v9.CheckScriptFailure(['def Func3(items: list)', 'echo "a"', 'enddef'], 'E1008: Missing <type> after list')
Bram Moolenaar62aec932022-01-29 21:45:34 +00002216 v9.CheckScriptFailure(['def Func4(...)', 'echo "a"', 'enddef'], 'E1055: Missing name after ...')
2217 v9.CheckScriptFailure(['def Func5(items:string)', 'echo "a"'], 'E1069:')
2218 v9.CheckScriptFailure(['def Func5(items)', 'echo "a"'], 'E1077:')
2219 v9.CheckScriptFailure(['def Func6(...x:list<number>)', 'echo "a"', 'enddef'], 'E1069:')
2220 v9.CheckScriptFailure(['def Func7(...x: int)', 'echo "a"', 'enddef'], 'E1010:')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002221enddef
2222
Bram Moolenaar86cdb8a2021-04-06 19:01:03 +02002223def Test_white_space_before_comma()
2224 var lines =<< trim END
2225 vim9script
2226 def Func(a: number , b: number)
2227 enddef
2228 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002229 v9.CheckScriptFailure(lines, 'E1068:')
Yegappan Lakshmanan611728f2021-05-24 15:15:47 +02002230 call assert_fails('vim9cmd echo stridx("a" .. "b" , "a")', 'E1068:')
Bram Moolenaar86cdb8a2021-04-06 19:01:03 +02002231enddef
2232
Bram Moolenaar608d78f2021-03-06 22:33:12 +01002233def Test_white_space_after_comma()
2234 var lines =<< trim END
2235 vim9script
2236 def Func(a: number,b: number)
2237 enddef
2238 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002239 v9.CheckScriptFailure(lines, 'E1069:')
Bram Moolenaar608d78f2021-03-06 22:33:12 +01002240
2241 # OK in legacy function
2242 lines =<< trim END
2243 vim9script
2244 func Func(a,b)
2245 endfunc
2246 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002247 v9.CheckScriptSuccess(lines)
Bram Moolenaar608d78f2021-03-06 22:33:12 +01002248enddef
2249
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002250def Test_vim9script_call()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002251 var lines =<< trim END
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002252 vim9script
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002253 var name = ''
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002254 def MyFunc(arg: string)
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002255 name = arg
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002256 enddef
2257 MyFunc('foobar')
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002258 name->assert_equal('foobar')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002259
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002260 var str = 'barfoo'
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002261 str->MyFunc()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002262 name->assert_equal('barfoo')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002263
Bram Moolenaar67979662020-06-20 22:50:47 +02002264 g:value = 'value'
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002265 g:value->MyFunc()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002266 name->assert_equal('value')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002267
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002268 var listvar = []
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002269 def ListFunc(arg: list<number>)
2270 listvar = arg
2271 enddef
2272 [1, 2, 3]->ListFunc()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002273 listvar->assert_equal([1, 2, 3])
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002274
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002275 var dictvar = {}
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002276 def DictFunc(arg: dict<number>)
2277 dictvar = arg
2278 enddef
Bram Moolenaare0de1712020-12-02 17:36:54 +01002279 {a: 1, b: 2}->DictFunc()
2280 dictvar->assert_equal({a: 1, b: 2})
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002281 def CompiledDict()
Bram Moolenaare0de1712020-12-02 17:36:54 +01002282 {a: 3, b: 4}->DictFunc()
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002283 enddef
2284 CompiledDict()
Bram Moolenaare0de1712020-12-02 17:36:54 +01002285 dictvar->assert_equal({a: 3, b: 4})
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002286
Bram Moolenaare0de1712020-12-02 17:36:54 +01002287 {a: 3, b: 4}->DictFunc()
2288 dictvar->assert_equal({a: 3, b: 4})
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002289
2290 ('text')->MyFunc()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002291 name->assert_equal('text')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002292 ("some")->MyFunc()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002293 name->assert_equal('some')
Bram Moolenaare6b53242020-07-01 17:28:33 +02002294
Bram Moolenaar13e12b82020-07-24 18:47:22 +02002295 # line starting with single quote is not a mark
Bram Moolenaar10409562020-07-29 20:00:38 +02002296 # line starting with double quote can be a method call
Bram Moolenaar3d48e252020-07-15 14:15:52 +02002297 'asdfasdf'->MyFunc()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002298 name->assert_equal('asdfasdf')
Bram Moolenaar10409562020-07-29 20:00:38 +02002299 "xyz"->MyFunc()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002300 name->assert_equal('xyz')
Bram Moolenaar3d48e252020-07-15 14:15:52 +02002301
2302 def UseString()
2303 'xyork'->MyFunc()
2304 enddef
2305 UseString()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002306 name->assert_equal('xyork')
Bram Moolenaar3d48e252020-07-15 14:15:52 +02002307
Bram Moolenaar10409562020-07-29 20:00:38 +02002308 def UseString2()
2309 "knife"->MyFunc()
2310 enddef
2311 UseString2()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002312 name->assert_equal('knife')
Bram Moolenaar10409562020-07-29 20:00:38 +02002313
Bram Moolenaar13e12b82020-07-24 18:47:22 +02002314 # prepending a colon makes it a mark
2315 new
2316 setline(1, ['aaa', 'bbb', 'ccc'])
2317 normal! 3Gmt1G
2318 :'t
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002319 getcurpos()[1]->assert_equal(3)
Bram Moolenaar13e12b82020-07-24 18:47:22 +02002320 bwipe!
2321
Bram Moolenaare6b53242020-07-01 17:28:33 +02002322 MyFunc(
2323 'continued'
2324 )
2325 assert_equal('continued',
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002326 name
Bram Moolenaare6b53242020-07-01 17:28:33 +02002327 )
2328
2329 call MyFunc(
2330 'more'
2331 ..
2332 'lines'
2333 )
2334 assert_equal(
2335 'morelines',
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002336 name)
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002337 END
Bram Moolenaarf5fec052022-09-11 11:49:22 +01002338 writefile(lines, 'Xcall.vim', 'D')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002339 source Xcall.vim
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002340enddef
2341
2342def Test_vim9script_call_fail_decl()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002343 var lines =<< trim END
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002344 vim9script
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002345 var name = ''
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002346 def MyFunc(arg: string)
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002347 var name = 123
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002348 enddef
Bram Moolenaar822ba242020-05-24 23:00:18 +02002349 defcompile
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002350 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002351 v9.CheckScriptFailure(lines, 'E1054:')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002352enddef
2353
Bram Moolenaar65b95452020-07-19 14:03:09 +02002354def Test_vim9script_call_fail_type()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002355 var lines =<< trim END
Bram Moolenaar65b95452020-07-19 14:03:09 +02002356 vim9script
2357 def MyFunc(arg: string)
2358 echo arg
2359 enddef
2360 MyFunc(1234)
2361 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002362 v9.CheckScriptFailure(lines, 'E1013: Argument 1: type mismatch, expected string but got number')
Bram Moolenaar65b95452020-07-19 14:03:09 +02002363enddef
2364
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002365def Test_vim9script_call_fail_const()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002366 var lines =<< trim END
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002367 vim9script
2368 const var = ''
2369 def MyFunc(arg: string)
2370 var = 'asdf'
2371 enddef
Bram Moolenaar822ba242020-05-24 23:00:18 +02002372 defcompile
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002373 END
Bram Moolenaarf5fec052022-09-11 11:49:22 +01002374 writefile(lines, 'Xcall_const.vim', 'D')
Bram Moolenaar9bd5d872020-09-06 21:47:48 +02002375 assert_fails('source Xcall_const.vim', 'E46:', '', 1, 'MyFunc')
Bram Moolenaar3bdc90b2020-12-22 20:35:40 +01002376
2377 lines =<< trim END
2378 const g:Aconst = 77
2379 def Change()
2380 # comment
2381 g:Aconst = 99
2382 enddef
2383 call Change()
2384 unlet g:Aconst
2385 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002386 v9.CheckScriptFailure(lines, 'E741: Value is locked: Aconst', 2)
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002387enddef
2388
2389" Test that inside :function a Python function can be defined, :def is not
2390" recognized.
2391func Test_function_python()
2392 CheckFeature python3
Bram Moolenaar727345e2020-09-27 23:33:59 +02002393 let py = 'python3'
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002394 execute py "<< EOF"
2395def do_something():
2396 return 1
2397EOF
2398endfunc
2399
2400def Test_delfunc()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002401 var lines =<< trim END
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002402 vim9script
Bram Moolenaar4c17ad92020-04-27 22:47:51 +02002403 def g:GoneSoon()
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002404 echo 'hello'
2405 enddef
2406
2407 def CallGoneSoon()
Bram Moolenaar62aec932022-01-29 21:45:34 +00002408 g:GoneSoon()
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002409 enddef
Bram Moolenaar822ba242020-05-24 23:00:18 +02002410 defcompile
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002411
Bram Moolenaar4c17ad92020-04-27 22:47:51 +02002412 delfunc g:GoneSoon
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002413 CallGoneSoon()
2414 END
Bram Moolenaarf5fec052022-09-11 11:49:22 +01002415 writefile(lines, 'XToDelFunc', 'D')
Bram Moolenaar9bd5d872020-09-06 21:47:48 +02002416 assert_fails('so XToDelFunc', 'E933:', '', 1, 'CallGoneSoon')
2417 assert_fails('so XToDelFunc', 'E933:', '', 1, 'CallGoneSoon')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002418enddef
2419
Bram Moolenaar7509ad82021-12-14 18:14:37 +00002420func Test_free_dict_while_in_funcstack()
2421 " relies on the sleep command
2422 CheckUnix
2423 call Run_Test_free_dict_while_in_funcstack()
2424endfunc
2425
2426def Run_Test_free_dict_while_in_funcstack()
Bram Moolenaar7509ad82021-12-14 18:14:37 +00002427 # this was freeing the TermRun() default argument dictionary while it was
2428 # still referenced in a funcstack_T
2429 var lines =<< trim END
2430 vim9script
2431
2432 &updatetime = 400
2433 def TermRun(_ = {})
2434 def Post()
2435 enddef
2436 def Exec()
2437 term_start('sleep 1', {
2438 term_finish: 'close',
2439 exit_cb: (_, _) => Post(),
2440 })
2441 enddef
2442 Exec()
2443 enddef
2444 nnoremap <F4> <Cmd>call <SID>TermRun()<CR>
2445 timer_start(100, (_) => feedkeys("\<F4>"))
2446 timer_start(1000, (_) => feedkeys("\<F4>"))
2447 sleep 1500m
2448 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002449 v9.CheckScriptSuccess(lines)
Bram Moolenaar7509ad82021-12-14 18:14:37 +00002450 nunmap <F4>
2451 set updatetime&
2452enddef
2453
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002454def Test_redef_failure()
Bram Moolenaard2c61702020-09-06 15:58:36 +02002455 writefile(['def Func0(): string', 'return "Func0"', 'enddef'], 'Xdef')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002456 so Xdef
Bram Moolenaard2c61702020-09-06 15:58:36 +02002457 writefile(['def Func1(): string', 'return "Func1"', 'enddef'], 'Xdef')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002458 so Xdef
Bram Moolenaard2c61702020-09-06 15:58:36 +02002459 writefile(['def! Func0(): string', 'enddef', 'defcompile'], 'Xdef')
Bram Moolenaar9bd5d872020-09-06 21:47:48 +02002460 assert_fails('so Xdef', 'E1027:', '', 1, 'Func0')
Bram Moolenaard2c61702020-09-06 15:58:36 +02002461 writefile(['def Func2(): string', 'return "Func2"', 'enddef'], 'Xdef')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002462 so Xdef
Bram Moolenaard2c61702020-09-06 15:58:36 +02002463 delete('Xdef')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002464
Bram Moolenaar701cc6c2021-04-10 13:33:48 +02002465 assert_fails('g:Func0()', 'E1091:')
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002466 g:Func1()->assert_equal('Func1')
2467 g:Func2()->assert_equal('Func2')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002468
2469 delfunc! Func0
2470 delfunc! Func1
2471 delfunc! Func2
2472enddef
2473
Bram Moolenaarf93c7fe2020-04-23 22:16:53 +02002474def Test_vim9script_func()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002475 var lines =<< trim END
Bram Moolenaarf93c7fe2020-04-23 22:16:53 +02002476 vim9script
2477 func Func(arg)
2478 echo a:arg
2479 endfunc
2480 Func('text')
2481 END
Bram Moolenaarf5fec052022-09-11 11:49:22 +01002482 writefile(lines, 'XVim9Func', 'D')
Bram Moolenaarf93c7fe2020-04-23 22:16:53 +02002483 so XVim9Func
Bram Moolenaarf93c7fe2020-04-23 22:16:53 +02002484enddef
2485
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002486let s:funcResult = 0
2487
2488def FuncNoArgNoRet()
Bram Moolenaar53900992020-08-22 19:02:02 +02002489 s:funcResult = 11
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002490enddef
2491
2492def FuncNoArgRetNumber(): number
Bram Moolenaar53900992020-08-22 19:02:02 +02002493 s:funcResult = 22
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002494 return 1234
2495enddef
2496
Bram Moolenaarec5929d2020-04-07 20:53:39 +02002497def FuncNoArgRetString(): string
Bram Moolenaar53900992020-08-22 19:02:02 +02002498 s:funcResult = 45
Bram Moolenaarec5929d2020-04-07 20:53:39 +02002499 return 'text'
2500enddef
2501
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002502def FuncOneArgNoRet(arg: number)
Bram Moolenaar53900992020-08-22 19:02:02 +02002503 s:funcResult = arg
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002504enddef
2505
2506def FuncOneArgRetNumber(arg: number): number
Bram Moolenaar53900992020-08-22 19:02:02 +02002507 s:funcResult = arg
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002508 return arg
2509enddef
2510
Bram Moolenaar08938ee2020-04-11 23:17:17 +02002511def FuncTwoArgNoRet(one: bool, two: number)
Bram Moolenaar53900992020-08-22 19:02:02 +02002512 s:funcResult = two
Bram Moolenaar08938ee2020-04-11 23:17:17 +02002513enddef
2514
Bram Moolenaar62aec932022-01-29 21:45:34 +00002515def s:FuncOneArgRetString(arg: string): string
Bram Moolenaarec5929d2020-04-07 20:53:39 +02002516 return arg
2517enddef
2518
Bram Moolenaar62aec932022-01-29 21:45:34 +00002519def s:FuncOneArgRetAny(arg: any): any
Bram Moolenaar89228602020-04-05 22:14:54 +02002520 return arg
2521enddef
2522
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002523def Test_func_type()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002524 var Ref1: func()
Bram Moolenaar53900992020-08-22 19:02:02 +02002525 s:funcResult = 0
Bram Moolenaar62aec932022-01-29 21:45:34 +00002526 Ref1 = g:FuncNoArgNoRet
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002527 Ref1()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002528 s:funcResult->assert_equal(11)
Bram Moolenaar4c683752020-04-05 21:38:23 +02002529
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002530 var Ref2: func
Bram Moolenaar53900992020-08-22 19:02:02 +02002531 s:funcResult = 0
Bram Moolenaar62aec932022-01-29 21:45:34 +00002532 Ref2 = g:FuncNoArgNoRet
Bram Moolenaar4c683752020-04-05 21:38:23 +02002533 Ref2()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002534 s:funcResult->assert_equal(11)
Bram Moolenaar4c683752020-04-05 21:38:23 +02002535
Bram Moolenaar53900992020-08-22 19:02:02 +02002536 s:funcResult = 0
Bram Moolenaar62aec932022-01-29 21:45:34 +00002537 Ref2 = g:FuncOneArgNoRet
Bram Moolenaar4c683752020-04-05 21:38:23 +02002538 Ref2(12)
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002539 s:funcResult->assert_equal(12)
Bram Moolenaar4c683752020-04-05 21:38:23 +02002540
Bram Moolenaar53900992020-08-22 19:02:02 +02002541 s:funcResult = 0
Bram Moolenaar62aec932022-01-29 21:45:34 +00002542 Ref2 = g:FuncNoArgRetNumber
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002543 Ref2()->assert_equal(1234)
2544 s:funcResult->assert_equal(22)
Bram Moolenaar4c683752020-04-05 21:38:23 +02002545
Bram Moolenaar53900992020-08-22 19:02:02 +02002546 s:funcResult = 0
Bram Moolenaar848fadd2022-01-30 15:28:30 +00002547 Ref2 = g:FuncOneArgRetNumber
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002548 Ref2(13)->assert_equal(13)
2549 s:funcResult->assert_equal(13)
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002550enddef
2551
Bram Moolenaar9978d472020-07-05 16:01:56 +02002552def Test_repeat_return_type()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002553 var res = 0
Bram Moolenaar9978d472020-07-05 16:01:56 +02002554 for n in repeat([1], 3)
2555 res += n
2556 endfor
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002557 res->assert_equal(3)
Bram Moolenaarfce82b32020-07-05 16:07:21 +02002558
2559 res = 0
Bakudankun375141e2022-09-09 18:46:47 +01002560 for n in repeat(0z01, 3)->blob2list()
2561 res += n
2562 endfor
2563 res->assert_equal(3)
2564
2565 res = 0
Bram Moolenaarfce82b32020-07-05 16:07:21 +02002566 for n in add([1, 2], 3)
2567 res += n
2568 endfor
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002569 res->assert_equal(6)
Bram Moolenaar9978d472020-07-05 16:01:56 +02002570enddef
2571
Bram Moolenaar846178a2020-07-05 17:04:13 +02002572def Test_argv_return_type()
2573 next fileone filetwo
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002574 var res = ''
Bram Moolenaar846178a2020-07-05 17:04:13 +02002575 for name in argv()
2576 res ..= name
2577 endfor
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002578 res->assert_equal('fileonefiletwo')
Bram Moolenaar846178a2020-07-05 17:04:13 +02002579enddef
2580
Bram Moolenaarec5929d2020-04-07 20:53:39 +02002581def Test_func_type_part()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002582 var RefVoid: func: void
Bram Moolenaar62aec932022-01-29 21:45:34 +00002583 RefVoid = g:FuncNoArgNoRet
2584 RefVoid = g:FuncOneArgNoRet
2585 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 +00002586 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 +02002587
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002588 var RefAny: func(): any
Bram Moolenaar62aec932022-01-29 21:45:34 +00002589 RefAny = g:FuncNoArgRetNumber
Bram Moolenaar848fadd2022-01-30 15:28:30 +00002590 RefAny = g:FuncNoArgRetString
Bram Moolenaar62aec932022-01-29 21:45:34 +00002591 v9.CheckDefFailure(['var RefAny: func(): any', 'RefAny = g:FuncNoArgNoRet'], 'E1012: Type mismatch; expected func(): any but got func()')
2592 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 +02002593
Bram Moolenaar6abd3dc2020-10-04 14:17:32 +02002594 var RefAnyNoArgs: func: any = RefAny
2595
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002596 var RefNr: func: number
Bram Moolenaar62aec932022-01-29 21:45:34 +00002597 RefNr = g:FuncNoArgRetNumber
Bram Moolenaar848fadd2022-01-30 15:28:30 +00002598 RefNr = g:FuncOneArgRetNumber
Bram Moolenaar62aec932022-01-29 21:45:34 +00002599 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 +00002600 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 +02002601
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002602 var RefStr: func: string
Bram Moolenaar848fadd2022-01-30 15:28:30 +00002603 RefStr = g:FuncNoArgRetString
Bram Moolenaarec5929d2020-04-07 20:53:39 +02002604 RefStr = FuncOneArgRetString
Bram Moolenaar62aec932022-01-29 21:45:34 +00002605 v9.CheckDefFailure(['var RefStr: func: string', 'RefStr = g:FuncNoArgNoRet'], 'E1012: Type mismatch; expected func(...): string but got func()')
2606 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 +02002607enddef
2608
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002609def Test_func_type_fails()
Bram Moolenaar62aec932022-01-29 21:45:34 +00002610 v9.CheckDefFailure(['var ref1: func()'], 'E704:')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002611
Bram Moolenaar62aec932022-01-29 21:45:34 +00002612 v9.CheckDefFailure(['var Ref1: func()', 'Ref1 = g:FuncNoArgRetNumber'], 'E1012: Type mismatch; expected func() but got func(): number')
2613 v9.CheckDefFailure(['var Ref1: func()', 'Ref1 = g:FuncOneArgNoRet'], 'E1012: Type mismatch; expected func() but got func(number)')
Bram Moolenaar848fadd2022-01-30 15:28:30 +00002614 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 +00002615 v9.CheckDefFailure(['var Ref1: func(bool)', 'Ref1 = g:FuncTwoArgNoRet'], 'E1012: Type mismatch; expected func(bool) but got func(bool, number)')
2616 v9.CheckDefFailure(['var Ref1: func(?bool)', 'Ref1 = g:FuncTwoArgNoRet'], 'E1012: Type mismatch; expected func(?bool) but got func(bool, number)')
2617 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 +02002618
Bram Moolenaar62aec932022-01-29 21:45:34 +00002619 v9.CheckDefFailure(['var RefWrong: func(string ,number)'], 'E1068:')
2620 v9.CheckDefFailure(['var RefWrong: func(string,number)'], 'E1069:')
2621 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:')
2622 v9.CheckDefFailure(['var RefWrong: func(bool):string'], 'E1069:')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002623enddef
2624
Bram Moolenaar89228602020-04-05 22:14:54 +02002625def Test_func_return_type()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002626 var nr: number
Bram Moolenaar62aec932022-01-29 21:45:34 +00002627 nr = g:FuncNoArgRetNumber()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002628 nr->assert_equal(1234)
Bram Moolenaar89228602020-04-05 22:14:54 +02002629
2630 nr = FuncOneArgRetAny(122)
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002631 nr->assert_equal(122)
Bram Moolenaar89228602020-04-05 22:14:54 +02002632
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002633 var str: string
Bram Moolenaar89228602020-04-05 22:14:54 +02002634 str = FuncOneArgRetAny('yes')
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002635 str->assert_equal('yes')
Bram Moolenaar89228602020-04-05 22:14:54 +02002636
Bram Moolenaar62aec932022-01-29 21:45:34 +00002637 v9.CheckDefFailure(['var str: string', 'str = g:FuncNoArgRetNumber()'], 'E1012: Type mismatch; expected string but got number')
Bram Moolenaar89228602020-04-05 22:14:54 +02002638enddef
2639
Bram Moolenaar6abd3dc2020-10-04 14:17:32 +02002640def Test_func_common_type()
2641 def FuncOne(n: number): number
2642 return n
2643 enddef
2644 def FuncTwo(s: string): number
2645 return len(s)
2646 enddef
2647 def FuncThree(n: number, s: string): number
2648 return n + len(s)
2649 enddef
2650 var list = [FuncOne, FuncTwo, FuncThree]
2651 assert_equal(8, list[0](8))
2652 assert_equal(4, list[1]('word'))
2653 assert_equal(7, list[2](3, 'word'))
2654enddef
2655
Bram Moolenaar62aec932022-01-29 21:45:34 +00002656def s:MultiLine(
Bram Moolenaar5e774c72020-04-12 21:53:00 +02002657 arg1: string,
2658 arg2 = 1234,
2659 ...rest: list<string>
2660 ): string
2661 return arg1 .. arg2 .. join(rest, '-')
2662enddef
2663
Bram Moolenaar2c330432020-04-13 14:41:35 +02002664def MultiLineComment(
2665 arg1: string, # comment
2666 arg2 = 1234, # comment
2667 ...rest: list<string> # comment
2668 ): string # comment
2669 return arg1 .. arg2 .. join(rest, '-')
2670enddef
2671
Bram Moolenaar5e774c72020-04-12 21:53:00 +02002672def Test_multiline()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002673 MultiLine('text')->assert_equal('text1234')
2674 MultiLine('text', 777)->assert_equal('text777')
2675 MultiLine('text', 777, 'one')->assert_equal('text777one')
2676 MultiLine('text', 777, 'one', 'two')->assert_equal('text777one-two')
Bram Moolenaar5e774c72020-04-12 21:53:00 +02002677enddef
2678
Bram Moolenaar23e03252020-04-12 22:22:31 +02002679func Test_multiline_not_vim9()
Bram Moolenaar62aec932022-01-29 21:45:34 +00002680 call s:MultiLine('text')->assert_equal('text1234')
2681 call s:MultiLine('text', 777)->assert_equal('text777')
2682 call s:MultiLine('text', 777, 'one')->assert_equal('text777one')
2683 call s:MultiLine('text', 777, 'one', 'two')->assert_equal('text777one-two')
Bram Moolenaar23e03252020-04-12 22:22:31 +02002684endfunc
2685
Bram Moolenaar5e774c72020-04-12 21:53:00 +02002686
Bram Moolenaaree4e0c12020-04-06 21:35:05 +02002687" When using CheckScriptFailure() for the below test, E1010 is generated instead
2688" of E1056.
2689func Test_E1056_1059()
2690 let caught_1056 = 0
2691 try
2692 def F():
2693 return 1
2694 enddef
2695 catch /E1056:/
2696 let caught_1056 = 1
2697 endtry
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002698 eval caught_1056->assert_equal(1)
Bram Moolenaaree4e0c12020-04-06 21:35:05 +02002699
2700 let caught_1059 = 0
2701 try
2702 def F5(items : list)
2703 echo 'a'
2704 enddef
2705 catch /E1059:/
2706 let caught_1059 = 1
2707 endtry
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002708 eval caught_1059->assert_equal(1)
Bram Moolenaaree4e0c12020-04-06 21:35:05 +02002709endfunc
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002710
Bram Moolenaar015f4262020-05-05 21:25:22 +02002711func DelMe()
2712 echo 'DelMe'
2713endfunc
2714
Bram Moolenaarbf8feb52020-08-08 14:26:31 +02002715def Test_error_reporting()
2716 # comment lines at the start of the function
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002717 var lines =<< trim END
Bram Moolenaarbf8feb52020-08-08 14:26:31 +02002718 " comment
2719 def Func()
2720 # comment
2721 # comment
2722 invalid
2723 enddef
2724 defcompile
2725 END
Bram Moolenaarf5fec052022-09-11 11:49:22 +01002726 writefile(lines, 'Xdef', 'D')
Bram Moolenaarbf8feb52020-08-08 14:26:31 +02002727 try
2728 source Xdef
Bram Moolenaar7517ffd2020-08-14 18:35:07 +02002729 assert_report('should have failed')
Bram Moolenaarbf8feb52020-08-08 14:26:31 +02002730 catch /E476:/
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002731 v:exception->assert_match('Invalid command: invalid')
2732 v:throwpoint->assert_match(', line 3$')
Bram Moolenaarbf8feb52020-08-08 14:26:31 +02002733 endtry
Bram Moolenaar2d870f82020-12-05 13:41:01 +01002734 delfunc! g:Func
Bram Moolenaarbf8feb52020-08-08 14:26:31 +02002735
2736 # comment lines after the start of the function
2737 lines =<< trim END
2738 " comment
2739 def Func()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002740 var x = 1234
Bram Moolenaarbf8feb52020-08-08 14:26:31 +02002741 # comment
2742 # comment
2743 invalid
2744 enddef
2745 defcompile
2746 END
Bram Moolenaar08052222020-09-14 17:04:31 +02002747 writefile(lines, 'Xdef')
Bram Moolenaarbf8feb52020-08-08 14:26:31 +02002748 try
2749 source Xdef
Bram Moolenaar7517ffd2020-08-14 18:35:07 +02002750 assert_report('should have failed')
Bram Moolenaarbf8feb52020-08-08 14:26:31 +02002751 catch /E476:/
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002752 v:exception->assert_match('Invalid command: invalid')
2753 v:throwpoint->assert_match(', line 4$')
Bram Moolenaarbf8feb52020-08-08 14:26:31 +02002754 endtry
Bram Moolenaar2d870f82020-12-05 13:41:01 +01002755 delfunc! g:Func
Bram Moolenaarbf8feb52020-08-08 14:26:31 +02002756
Bram Moolenaar7517ffd2020-08-14 18:35:07 +02002757 lines =<< trim END
2758 vim9script
2759 def Func()
Bram Moolenaare0de1712020-12-02 17:36:54 +01002760 var db = {foo: 1, bar: 2}
Bram Moolenaar7517ffd2020-08-14 18:35:07 +02002761 # comment
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002762 var x = db.asdf
Bram Moolenaar7517ffd2020-08-14 18:35:07 +02002763 enddef
2764 defcompile
2765 Func()
2766 END
Bram Moolenaar08052222020-09-14 17:04:31 +02002767 writefile(lines, 'Xdef')
Bram Moolenaar7517ffd2020-08-14 18:35:07 +02002768 try
2769 source Xdef
2770 assert_report('should have failed')
2771 catch /E716:/
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002772 v:throwpoint->assert_match('_Func, line 3$')
Bram Moolenaar7517ffd2020-08-14 18:35:07 +02002773 endtry
Bram Moolenaar2d870f82020-12-05 13:41:01 +01002774 delfunc! g:Func
Bram Moolenaarbf8feb52020-08-08 14:26:31 +02002775enddef
2776
Bram Moolenaar015f4262020-05-05 21:25:22 +02002777def Test_deleted_function()
Bram Moolenaar62aec932022-01-29 21:45:34 +00002778 v9.CheckDefExecFailure([
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002779 'var RefMe: func = function("g:DelMe")',
Bram Moolenaar015f4262020-05-05 21:25:22 +02002780 'delfunc g:DelMe',
2781 'echo RefMe()'], 'E117:')
2782enddef
2783
2784def Test_unknown_function()
Bram Moolenaar62aec932022-01-29 21:45:34 +00002785 v9.CheckDefExecFailure([
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002786 'var Ref: func = function("NotExist")',
Bram Moolenaar9b7bf9e2020-07-11 22:14:59 +02002787 'delfunc g:NotExist'], 'E700:')
Bram Moolenaar015f4262020-05-05 21:25:22 +02002788enddef
2789
Bram Moolenaar62aec932022-01-29 21:45:34 +00002790def s:RefFunc(Ref: func(any): any): string
Bram Moolenaarc8cd2b32020-05-01 19:29:08 +02002791 return Ref('more')
2792enddef
2793
2794def Test_closure_simple()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002795 var local = 'some '
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01002796 RefFunc((s) => local .. s)->assert_equal('some more')
Bram Moolenaarc8cd2b32020-05-01 19:29:08 +02002797enddef
2798
Bram Moolenaar62aec932022-01-29 21:45:34 +00002799def s:MakeRef()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002800 var local = 'some '
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01002801 g:Ref = (s) => local .. s
Bram Moolenaarbf67ea12020-05-02 17:52:42 +02002802enddef
2803
2804def Test_closure_ref_after_return()
2805 MakeRef()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002806 g:Ref('thing')->assert_equal('some thing')
Bram Moolenaarbf67ea12020-05-02 17:52:42 +02002807 unlet g:Ref
2808enddef
2809
Bram Moolenaar62aec932022-01-29 21:45:34 +00002810def s:MakeTwoRefs()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002811 var local = ['some']
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01002812 g:Extend = (s) => local->add(s)
2813 g:Read = () => local
Bram Moolenaar5adc55c2020-05-02 23:12:58 +02002814enddef
2815
2816def Test_closure_two_refs()
2817 MakeTwoRefs()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002818 join(g:Read(), ' ')->assert_equal('some')
Bram Moolenaar5adc55c2020-05-02 23:12:58 +02002819 g:Extend('more')
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002820 join(g:Read(), ' ')->assert_equal('some more')
Bram Moolenaar5adc55c2020-05-02 23:12:58 +02002821 g:Extend('even')
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002822 join(g:Read(), ' ')->assert_equal('some more even')
Bram Moolenaar5adc55c2020-05-02 23:12:58 +02002823
2824 unlet g:Extend
2825 unlet g:Read
2826enddef
2827
Bram Moolenaar62aec932022-01-29 21:45:34 +00002828def s:ReadRef(Ref: func(): list<string>): string
Bram Moolenaar5adc55c2020-05-02 23:12:58 +02002829 return join(Ref(), ' ')
2830enddef
2831
Bram Moolenaar62aec932022-01-29 21:45:34 +00002832def s:ExtendRef(Ref: func(string): list<string>, add: string)
Bram Moolenaar5adc55c2020-05-02 23:12:58 +02002833 Ref(add)
2834enddef
2835
2836def Test_closure_two_indirect_refs()
Bram Moolenaarf7779c62020-05-03 15:38:16 +02002837 MakeTwoRefs()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002838 ReadRef(g:Read)->assert_equal('some')
Bram Moolenaar5adc55c2020-05-02 23:12:58 +02002839 ExtendRef(g:Extend, 'more')
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002840 ReadRef(g:Read)->assert_equal('some more')
Bram Moolenaar5adc55c2020-05-02 23:12:58 +02002841 ExtendRef(g:Extend, 'even')
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002842 ReadRef(g:Read)->assert_equal('some more even')
Bram Moolenaar5adc55c2020-05-02 23:12:58 +02002843
2844 unlet g:Extend
2845 unlet g:Read
2846enddef
Bram Moolenaarbf67ea12020-05-02 17:52:42 +02002847
Bram Moolenaar62aec932022-01-29 21:45:34 +00002848def s:MakeArgRefs(theArg: string)
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002849 var local = 'loc_val'
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01002850 g:UseArg = (s) => theArg .. '/' .. local .. '/' .. s
Bram Moolenaar2fd4cd72020-05-03 22:30:49 +02002851enddef
2852
Bram Moolenaar62aec932022-01-29 21:45:34 +00002853def s:MakeArgRefsVarargs(theArg: string, ...rest: list<string>)
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002854 var local = 'the_loc'
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01002855 g:UseVararg = (s) => theArg .. '/' .. local .. '/' .. s .. '/' .. join(rest)
Bram Moolenaar2fd4cd72020-05-03 22:30:49 +02002856enddef
2857
2858def Test_closure_using_argument()
2859 MakeArgRefs('arg_val')
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002860 g:UseArg('call_val')->assert_equal('arg_val/loc_val/call_val')
Bram Moolenaar2fd4cd72020-05-03 22:30:49 +02002861
2862 MakeArgRefsVarargs('arg_val', 'one', 'two')
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002863 g:UseVararg('call_val')->assert_equal('arg_val/the_loc/call_val/one two')
Bram Moolenaar2fd4cd72020-05-03 22:30:49 +02002864
2865 unlet g:UseArg
2866 unlet g:UseVararg
Bram Moolenaar44ec21c2021-02-12 21:50:57 +01002867
2868 var lines =<< trim END
2869 vim9script
2870 def Test(Fun: func(number): number): list<number>
2871 return map([1, 2, 3], (_, i) => Fun(i))
2872 enddef
2873 def Inc(nr: number): number
2874 return nr + 2
2875 enddef
2876 assert_equal([3, 4, 5], Test(Inc))
2877 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002878 v9.CheckScriptSuccess(lines)
Bram Moolenaar2fd4cd72020-05-03 22:30:49 +02002879enddef
2880
Bram Moolenaar62aec932022-01-29 21:45:34 +00002881def s:MakeGetAndAppendRefs()
Bram Moolenaar85d5e2b2020-10-10 14:13:01 +02002882 var local = 'a'
2883
2884 def Append(arg: string)
2885 local ..= arg
2886 enddef
2887 g:Append = Append
2888
2889 def Get(): string
2890 return local
2891 enddef
2892 g:Get = Get
2893enddef
2894
2895def Test_closure_append_get()
2896 MakeGetAndAppendRefs()
2897 g:Get()->assert_equal('a')
2898 g:Append('-b')
2899 g:Get()->assert_equal('a-b')
2900 g:Append('-c')
2901 g:Get()->assert_equal('a-b-c')
2902
2903 unlet g:Append
2904 unlet g:Get
2905enddef
Bram Moolenaarb68b3462020-05-06 21:06:30 +02002906
Bram Moolenaar04b12692020-05-04 23:24:44 +02002907def Test_nested_closure()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002908 var local = 'text'
Bram Moolenaar04b12692020-05-04 23:24:44 +02002909 def Closure(arg: string): string
2910 return local .. arg
2911 enddef
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002912 Closure('!!!')->assert_equal('text!!!')
Bram Moolenaar04b12692020-05-04 23:24:44 +02002913enddef
2914
Bram Moolenaar62aec932022-01-29 21:45:34 +00002915func s:GetResult(Ref)
Bram Moolenaar6f5b6df2020-05-16 21:20:12 +02002916 return a:Ref('some')
2917endfunc
2918
2919def Test_call_closure_not_compiled()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002920 var text = 'text'
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01002921 g:Ref = (s) => s .. text
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002922 GetResult(g:Ref)->assert_equal('sometext')
Bram Moolenaar6f5b6df2020-05-16 21:20:12 +02002923enddef
2924
Bram Moolenaar7cbfaa52020-09-18 21:25:32 +02002925def Test_double_closure_fails()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002926 var lines =<< trim END
Bram Moolenaar7cbfaa52020-09-18 21:25:32 +02002927 vim9script
2928 def Func()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002929 var name = 0
2930 for i in range(2)
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01002931 timer_start(0, () => name)
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002932 endfor
Bram Moolenaar7cbfaa52020-09-18 21:25:32 +02002933 enddef
2934 Func()
2935 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002936 v9.CheckScriptSuccess(lines)
Bram Moolenaar7cbfaa52020-09-18 21:25:32 +02002937enddef
2938
Bram Moolenaar85d5e2b2020-10-10 14:13:01 +02002939def Test_nested_closure_used()
2940 var lines =<< trim END
2941 vim9script
2942 def Func()
2943 var x = 'hello'
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01002944 var Closure = () => x
2945 g:Myclosure = () => Closure()
Bram Moolenaar85d5e2b2020-10-10 14:13:01 +02002946 enddef
2947 Func()
2948 assert_equal('hello', g:Myclosure())
2949 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002950 v9.CheckScriptSuccess(lines)
Bram Moolenaar85d5e2b2020-10-10 14:13:01 +02002951enddef
Bram Moolenaar0876c782020-10-07 19:08:04 +02002952
Bram Moolenaarc70bdab2020-09-26 19:59:38 +02002953def Test_nested_closure_fails()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002954 var lines =<< trim END
Bram Moolenaarc70bdab2020-09-26 19:59:38 +02002955 vim9script
2956 def FuncA()
2957 FuncB(0)
2958 enddef
2959 def FuncB(n: number): list<string>
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01002960 return map([0], (_, v) => n)
Bram Moolenaarc70bdab2020-09-26 19:59:38 +02002961 enddef
2962 FuncA()
2963 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002964 v9.CheckScriptFailure(lines, 'E1012:')
Bram Moolenaarc70bdab2020-09-26 19:59:38 +02002965enddef
2966
Bram Moolenaar6de22962022-09-09 21:35:36 +01002967def Run_Test_closure_in_for_loop_fails()
2968 var lines =<< trim END
2969 vim9script
Bram Moolenaar766ae5b2022-09-14 00:30:51 +01002970 redraw
Bram Moolenaar6de22962022-09-09 21:35:36 +01002971 for n in [0]
Bram Moolenaar766ae5b2022-09-14 00:30:51 +01002972 # time should be enough for startup to finish
2973 timer_start(200, (_) => {
Bram Moolenaar6de22962022-09-09 21:35:36 +01002974 echo n
2975 })
2976 endfor
2977 END
2978 writefile(lines, 'XTest_closure_fails', 'D')
2979
2980 # Check that an error shows
Bram Moolenaarc069ede2022-09-11 12:01:04 +01002981 var buf = g:RunVimInTerminal('-S XTest_closure_fails', {rows: 6, wait_for_ruler: 0})
Bram Moolenaar766ae5b2022-09-14 00:30:51 +01002982 g:VerifyScreenDump(buf, 'Test_vim9_closure_fails', {wait: 3000})
Bram Moolenaar6de22962022-09-09 21:35:36 +01002983
2984 # clean up
2985 g:StopVimInTerminal(buf)
2986enddef
2987
2988func Test_closure_in_for_loop_fails()
2989 CheckScreendump
2990 call Run_Test_closure_in_for_loop_fails()
2991endfunc
2992
Bram Moolenaarf112f302020-12-20 17:47:52 +01002993def Test_global_closure()
2994 var lines =<< trim END
2995 vim9script
2996 def ReverseEveryNLines(n: number, line1: number, line2: number)
2997 var mods = 'sil keepj keepp lockm '
2998 var range = ':' .. line1 .. ',' .. line2
2999 def g:Offset(): number
3000 var offset = (line('.') - line1 + 1) % n
3001 return offset != 0 ? offset : n
3002 enddef
3003 exe mods .. range .. 'g/^/exe "m .-" .. g:Offset()'
3004 enddef
3005
3006 new
3007 repeat(['aaa', 'bbb', 'ccc'], 3)->setline(1)
3008 ReverseEveryNLines(3, 1, 9)
3009 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003010 v9.CheckScriptSuccess(lines)
Bram Moolenaarf112f302020-12-20 17:47:52 +01003011 var expected = repeat(['ccc', 'bbb', 'aaa'], 3)
3012 assert_equal(expected, getline(1, 9))
3013 bwipe!
3014enddef
3015
Bram Moolenaarcd45ed02020-12-22 17:35:54 +01003016def Test_global_closure_called_directly()
3017 var lines =<< trim END
3018 vim9script
3019 def Outer()
3020 var x = 1
3021 def g:Inner()
3022 var y = x
3023 x += 1
3024 assert_equal(1, y)
3025 enddef
3026 g:Inner()
3027 assert_equal(2, x)
3028 enddef
3029 Outer()
3030 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003031 v9.CheckScriptSuccess(lines)
Bram Moolenaarcd45ed02020-12-22 17:35:54 +01003032 delfunc g:Inner
3033enddef
3034
Bram Moolenaar69c76172021-12-02 16:38:52 +00003035def Test_closure_called_from_legacy()
3036 var lines =<< trim END
3037 vim9script
3038 def Func()
3039 var outer = 'foo'
3040 var F = () => {
3041 outer = 'bar'
3042 }
3043 execute printf('call %s()', string(F))
3044 enddef
3045 Func()
3046 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003047 v9.CheckScriptFailure(lines, 'E1248')
Bram Moolenaar69c76172021-12-02 16:38:52 +00003048enddef
3049
Bram Moolenaar34c54eb2020-11-25 19:15:19 +01003050def Test_failure_in_called_function()
3051 # this was using the frame index as the return value
3052 var lines =<< trim END
3053 vim9script
3054 au TerminalWinOpen * eval [][0]
3055 def PopupTerm(a: any)
3056 # make sure typvals on stack are string
3057 ['a', 'b', 'c', 'd', 'e', 'f', 'g']->join()
3058 FireEvent()
3059 enddef
3060 def FireEvent()
3061 do TerminalWinOpen
3062 enddef
3063 # use try/catch to make eval fail
3064 try
3065 call PopupTerm(0)
3066 catch
3067 endtry
3068 au! TerminalWinOpen
3069 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003070 v9.CheckScriptSuccess(lines)
Bram Moolenaar34c54eb2020-11-25 19:15:19 +01003071enddef
3072
Bram Moolenaar5366e1a2020-10-01 13:01:34 +02003073def Test_nested_lambda()
3074 var lines =<< trim END
3075 vim9script
3076 def Func()
3077 var x = 4
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01003078 var Lambda1 = () => 7
3079 var Lambda2 = () => [Lambda1(), x]
Bram Moolenaar5366e1a2020-10-01 13:01:34 +02003080 var res = Lambda2()
3081 assert_equal([7, 4], res)
3082 enddef
3083 Func()
3084 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003085 v9.CheckScriptSuccess(lines)
Bram Moolenaar5366e1a2020-10-01 13:01:34 +02003086enddef
3087
Bram Moolenaarc04f2a42021-06-09 19:30:03 +02003088def Test_double_nested_lambda()
3089 var lines =<< trim END
3090 vim9script
3091 def F(head: string): func(string): func(string): string
3092 return (sep: string): func(string): string => ((tail: string): string => {
3093 return head .. sep .. tail
3094 })
3095 enddef
3096 assert_equal('hello-there', F('hello')('-')('there'))
3097 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003098 v9.CheckScriptSuccess(lines)
Bram Moolenaarc04f2a42021-06-09 19:30:03 +02003099enddef
3100
Bram Moolenaar074f84c2021-05-18 11:47:44 +02003101def Test_nested_inline_lambda()
Bram Moolenaar074f84c2021-05-18 11:47:44 +02003102 var lines =<< trim END
3103 vim9script
3104 def F(text: string): func(string): func(string): string
3105 return (arg: string): func(string): string => ((sep: string): string => {
Bram Moolenaar23e2e112021-08-03 21:16:18 +02003106 return sep .. arg .. text
Bram Moolenaar074f84c2021-05-18 11:47:44 +02003107 })
3108 enddef
Bram Moolenaar23e2e112021-08-03 21:16:18 +02003109 assert_equal('--there++', F('++')('there')('--'))
Bram Moolenaar074f84c2021-05-18 11:47:44 +02003110 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003111 v9.CheckScriptSuccess(lines)
Bram Moolenaar5245beb2021-07-15 22:03:50 +02003112
3113 lines =<< trim END
3114 vim9script
3115 echo range(4)->mapnew((_, v) => {
3116 return range(v) ->mapnew((_, s) => {
3117 return string(s)
3118 })
3119 })
3120 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003121 v9.CheckScriptSuccess(lines)
Bram Moolenaarc6ba2f92021-07-18 13:42:29 +02003122
3123 lines =<< trim END
3124 vim9script
3125
Bram Moolenaara749a422022-02-12 19:52:25 +00003126 def Func()
Bram Moolenaarc6ba2f92021-07-18 13:42:29 +02003127 range(10)
3128 ->mapnew((_, _) => ({
3129 key: range(10)->mapnew((_, _) => {
3130 return ' '
3131 }),
3132 }))
3133 enddef
3134
3135 defcomp
3136 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003137 v9.CheckScriptSuccess(lines)
Bram Moolenaar074f84c2021-05-18 11:47:44 +02003138enddef
3139
Bram Moolenaar52bf81c2020-11-17 18:50:44 +01003140def Shadowed(): list<number>
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01003141 var FuncList: list<func: number> = [() => 42]
Bram Moolenaar75ab91f2021-01-10 22:42:50 +01003142 return FuncList->mapnew((_, Shadowed) => Shadowed())
Bram Moolenaar52bf81c2020-11-17 18:50:44 +01003143enddef
3144
3145def Test_lambda_arg_shadows_func()
Bram Moolenaar62aec932022-01-29 21:45:34 +00003146 assert_equal([42], g:Shadowed())
Bram Moolenaar52bf81c2020-11-17 18:50:44 +01003147enddef
3148
Bram Moolenaar21dc8f12022-03-16 17:54:17 +00003149def Test_compiling_referenced_func_no_shadow()
3150 var lines =<< trim END
3151 vim9script
3152
3153 def InitializeReply(lspserver: dict<any>)
3154 enddef
3155
3156 def ProcessReply(lspserver: dict<any>)
3157 var lsp_reply_handlers: dict<func> =
3158 { 'initialize': InitializeReply }
3159 lsp_reply_handlers['initialize'](lspserver)
3160 enddef
3161
3162 call ProcessReply({})
3163 END
3164 v9.CheckScriptSuccess(lines)
3165enddef
3166
Bram Moolenaar62aec932022-01-29 21:45:34 +00003167def s:Line_continuation_in_def(dir: string = ''): string
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02003168 var path: string = empty(dir)
3169 \ ? 'empty'
3170 \ : 'full'
3171 return path
Bram Moolenaaracd4c5e2020-06-22 19:39:03 +02003172enddef
3173
3174def Test_line_continuation_in_def()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02003175 Line_continuation_in_def('.')->assert_equal('full')
Bram Moolenaaracd4c5e2020-06-22 19:39:03 +02003176enddef
3177
Bram Moolenaar2ea95b62020-11-19 21:47:56 +01003178def Test_script_var_in_lambda()
3179 var lines =<< trim END
3180 vim9script
3181 var script = 'test'
Bram Moolenaarbb8a7ce2021-04-10 20:10:26 +02003182 assert_equal(['test'], map(['one'], (_, _) => script))
Bram Moolenaar2ea95b62020-11-19 21:47:56 +01003183 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003184 v9.CheckScriptSuccess(lines)
Bram Moolenaar2ea95b62020-11-19 21:47:56 +01003185enddef
3186
Bram Moolenaar62aec932022-01-29 21:45:34 +00003187def s:Line_continuation_in_lambda(): list<string>
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02003188 var x = range(97, 100)
Bram Moolenaar75ab91f2021-01-10 22:42:50 +01003189 ->mapnew((_, v) => nr2char(v)
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01003190 ->toupper())
Bram Moolenaar7a4b8982020-07-08 17:36:21 +02003191 ->reverse()
3192 return x
3193enddef
3194
3195def Test_line_continuation_in_lambda()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02003196 Line_continuation_in_lambda()->assert_equal(['D', 'C', 'B', 'A'])
Bram Moolenaarf898f7c2021-01-16 18:09:52 +01003197
3198 var lines =<< trim END
3199 vim9script
3200 var res = [{n: 1, m: 2, s: 'xxx'}]
3201 ->mapnew((_, v: dict<any>): string => printf('%d:%d:%s',
3202 v.n,
3203 v.m,
3204 substitute(v.s, '.*', 'yyy', '')
3205 ))
3206 assert_equal(['1:2:yyy'], res)
3207 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003208 v9.CheckScriptSuccess(lines)
Bram Moolenaar7a4b8982020-07-08 17:36:21 +02003209enddef
3210
Bram Moolenaarb6571982021-01-08 22:24:19 +01003211def Test_list_lambda()
3212 timer_start(1000, (_) => 0)
3213 var body = execute(timer_info()[0].callback
3214 ->string()
3215 ->substitute("('", ' ', '')
3216 ->substitute("')", '', '')
3217 ->substitute('function\zs', ' ', ''))
Bram Moolenaar767034c2021-04-09 17:24:52 +02003218 assert_match('def <lambda>\d\+(_: any): number\n1 return 0\n enddef', body)
Bram Moolenaarb6571982021-01-08 22:24:19 +01003219enddef
3220
Bram Moolenaar3c77b6a2021-07-25 18:07:00 +02003221def Test_lambda_block_variable()
Bram Moolenaar88421d62021-07-24 14:14:52 +02003222 var lines =<< trim END
3223 vim9script
3224 var flist: list<func>
3225 for i in range(10)
3226 var inloop = i
3227 flist[i] = () => inloop
3228 endfor
3229 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003230 v9.CheckScriptSuccess(lines)
Bram Moolenaar88421d62021-07-24 14:14:52 +02003231
3232 lines =<< trim END
3233 vim9script
3234 if true
3235 var outloop = 5
3236 var flist: list<func>
3237 for i in range(10)
3238 flist[i] = () => outloop
3239 endfor
3240 endif
3241 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003242 v9.CheckScriptSuccess(lines)
Bram Moolenaar88421d62021-07-24 14:14:52 +02003243
3244 lines =<< trim END
3245 vim9script
3246 if true
3247 var outloop = 5
3248 endif
3249 var flist: list<func>
3250 for i in range(10)
3251 flist[i] = () => outloop
3252 endfor
3253 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003254 v9.CheckScriptFailure(lines, 'E1001: Variable not found: outloop', 1)
Bram Moolenaar3c77b6a2021-07-25 18:07:00 +02003255
3256 lines =<< trim END
3257 vim9script
3258 for i in range(10)
3259 var Ref = () => 0
3260 endfor
3261 assert_equal(0, ((i) => 0)(0))
3262 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003263 v9.CheckScriptSuccess(lines)
Bram Moolenaar88421d62021-07-24 14:14:52 +02003264enddef
3265
Bram Moolenaar96cf4ba2021-04-24 14:15:41 +02003266def Test_legacy_lambda()
3267 legacy echo {x -> 'hello ' .. x}('foo')
Bram Moolenaardc4c2302021-04-25 13:54:42 +02003268
Bram Moolenaar96cf4ba2021-04-24 14:15:41 +02003269 var lines =<< trim END
3270 echo {x -> 'hello ' .. x}('foo')
3271 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003272 v9.CheckDefAndScriptFailure(lines, 'E720:')
Bram Moolenaardc4c2302021-04-25 13:54:42 +02003273
3274 lines =<< trim END
3275 vim9script
3276 def Func()
3277 echo (() => 'no error')()
3278 enddef
3279 legacy call s:Func()
3280 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003281 v9.CheckScriptSuccess(lines)
Bram Moolenaar96cf4ba2021-04-24 14:15:41 +02003282enddef
3283
Bram Moolenaarce024c32021-06-26 13:00:49 +02003284def Test_legacy()
3285 var lines =<< trim END
3286 vim9script
3287 func g:LegacyFunction()
3288 let g:legacyvar = 1
3289 endfunc
3290 def Testit()
3291 legacy call g:LegacyFunction()
3292 enddef
3293 Testit()
3294 assert_equal(1, g:legacyvar)
3295 unlet g:legacyvar
3296 delfunc g:LegacyFunction
3297 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003298 v9.CheckScriptSuccess(lines)
Bram Moolenaarce024c32021-06-26 13:00:49 +02003299enddef
3300
Bram Moolenaarc3cb1c92021-06-02 16:47:53 +02003301def Test_legacy_errors()
3302 for cmd in ['if', 'elseif', 'else', 'endif',
3303 'for', 'endfor', 'continue', 'break',
3304 'while', 'endwhile',
3305 'try', 'catch', 'finally', 'endtry']
Bram Moolenaar62aec932022-01-29 21:45:34 +00003306 v9.CheckDefFailure(['legacy ' .. cmd .. ' expr'], 'E1189:')
Bram Moolenaarc3cb1c92021-06-02 16:47:53 +02003307 endfor
3308enddef
3309
Bram Moolenaarb1b6f4d2021-09-13 18:25:54 +02003310def Test_call_legacy_with_dict()
3311 var lines =<< trim END
3312 vim9script
3313 func Legacy() dict
3314 let g:result = self.value
3315 endfunc
3316 def TestDirect()
3317 var d = {value: 'yes', func: Legacy}
3318 d.func()
3319 enddef
3320 TestDirect()
3321 assert_equal('yes', g:result)
3322 unlet g:result
3323
3324 def TestIndirect()
3325 var d = {value: 'foo', func: Legacy}
3326 var Fi = d.func
3327 Fi()
3328 enddef
3329 TestIndirect()
3330 assert_equal('foo', g:result)
3331 unlet g:result
3332
3333 var d = {value: 'bar', func: Legacy}
3334 d.func()
3335 assert_equal('bar', g:result)
3336 unlet g:result
3337 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003338 v9.CheckScriptSuccess(lines)
Bram Moolenaarb1b6f4d2021-09-13 18:25:54 +02003339enddef
3340
Bram Moolenaar62aec932022-01-29 21:45:34 +00003341def s:DoFilterThis(a: string): list<string>
Bram Moolenaarab360522021-01-10 14:02:28 +01003342 # closure nested inside another closure using argument
3343 var Filter = (l) => filter(l, (_, v) => stridx(v, a) == 0)
3344 return ['x', 'y', 'a', 'x2', 'c']->Filter()
3345enddef
3346
3347def Test_nested_closure_using_argument()
3348 assert_equal(['x', 'x2'], DoFilterThis('x'))
3349enddef
3350
Bram Moolenaar0186e582021-01-10 18:33:11 +01003351def Test_triple_nested_closure()
3352 var what = 'x'
3353 var Match = (val: string, cmp: string): bool => stridx(val, cmp) == 0
3354 var Filter = (l) => filter(l, (_, v) => Match(v, what))
3355 assert_equal(['x', 'x2'], ['x', 'y', 'a', 'x2', 'c']->Filter())
3356enddef
3357
Bram Moolenaar8f510af2020-07-05 18:48:23 +02003358func Test_silent_echo()
Bram Moolenaar47e7d702020-07-05 18:18:42 +02003359 CheckScreendump
Bram Moolenaar3b309f12021-12-13 18:19:55 +00003360 call Run_Test_silent_echo()
3361endfunc
Bram Moolenaar47e7d702020-07-05 18:18:42 +02003362
Bram Moolenaar3b309f12021-12-13 18:19:55 +00003363def Run_Test_silent_echo()
3364 var lines =<< trim END
Bram Moolenaar47e7d702020-07-05 18:18:42 +02003365 vim9script
3366 def EchoNothing()
3367 silent echo ''
3368 enddef
3369 defcompile
3370 END
Bram Moolenaar6de22962022-09-09 21:35:36 +01003371 writefile(lines, 'XTest_silent_echo', 'D')
Bram Moolenaar47e7d702020-07-05 18:18:42 +02003372
Bram Moolenaar3b309f12021-12-13 18:19:55 +00003373 # Check that the balloon shows up after a mouse move
Bram Moolenaar62aec932022-01-29 21:45:34 +00003374 var buf = g:RunVimInTerminal('-S XTest_silent_echo', {'rows': 6})
Bram Moolenaar3b309f12021-12-13 18:19:55 +00003375 term_sendkeys(buf, ":abc")
Bram Moolenaar62aec932022-01-29 21:45:34 +00003376 g:VerifyScreenDump(buf, 'Test_vim9_silent_echo', {})
Bram Moolenaar47e7d702020-07-05 18:18:42 +02003377
Bram Moolenaar3b309f12021-12-13 18:19:55 +00003378 # clean up
Bram Moolenaar62aec932022-01-29 21:45:34 +00003379 g:StopVimInTerminal(buf)
Bram Moolenaar3b309f12021-12-13 18:19:55 +00003380enddef
Bram Moolenaar47e7d702020-07-05 18:18:42 +02003381
Bram Moolenaar171fb922020-10-28 16:54:47 +01003382def SilentlyError()
3383 execute('silent! invalid')
3384 g:did_it = 'yes'
3385enddef
3386
Bram Moolenaar62aec932022-01-29 21:45:34 +00003387func s:UserError()
Bram Moolenaar28ee8922020-10-28 20:20:00 +01003388 silent! invalid
3389endfunc
3390
3391def SilentlyUserError()
3392 UserError()
3393 g:did_it = 'yes'
3394enddef
Bram Moolenaar171fb922020-10-28 16:54:47 +01003395
3396" This can't be a :def function, because the assert would not be reached.
Bram Moolenaar171fb922020-10-28 16:54:47 +01003397func Test_ignore_silent_error()
3398 let g:did_it = 'no'
3399 call SilentlyError()
3400 call assert_equal('yes', g:did_it)
3401
Bram Moolenaar28ee8922020-10-28 20:20:00 +01003402 let g:did_it = 'no'
3403 call SilentlyUserError()
3404 call assert_equal('yes', g:did_it)
Bram Moolenaar171fb922020-10-28 16:54:47 +01003405
3406 unlet g:did_it
3407endfunc
3408
Bram Moolenaarcd030c42020-10-30 21:49:40 +01003409def Test_ignore_silent_error_in_filter()
3410 var lines =<< trim END
3411 vim9script
3412 def Filter(winid: number, key: string): bool
3413 if key == 'o'
3414 silent! eval [][0]
3415 return true
3416 endif
3417 return popup_filter_menu(winid, key)
3418 enddef
3419
Bram Moolenaare0de1712020-12-02 17:36:54 +01003420 popup_create('popup', {filter: Filter})
Bram Moolenaarcd030c42020-10-30 21:49:40 +01003421 feedkeys("o\r", 'xnt')
3422 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003423 v9.CheckScriptSuccess(lines)
Bram Moolenaarcd030c42020-10-30 21:49:40 +01003424enddef
3425
Bram Moolenaar62aec932022-01-29 21:45:34 +00003426def s:Fibonacci(n: number): number
Bram Moolenaar4b9bd692020-09-05 21:57:53 +02003427 if n < 2
3428 return n
3429 else
3430 return Fibonacci(n - 1) + Fibonacci(n - 2)
3431 endif
3432enddef
3433
Bram Moolenaar985116a2020-07-12 17:31:09 +02003434def Test_recursive_call()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02003435 Fibonacci(20)->assert_equal(6765)
Bram Moolenaar985116a2020-07-12 17:31:09 +02003436enddef
3437
Bram Moolenaar62aec932022-01-29 21:45:34 +00003438def s:TreeWalk(dir: string): list<any>
Bram Moolenaar75ab91f2021-01-10 22:42:50 +01003439 return readdir(dir)->mapnew((_, val) =>
Bram Moolenaar08f7a412020-07-13 20:41:08 +02003440 fnamemodify(dir .. '/' .. val, ':p')->isdirectory()
Bram Moolenaar2bede172020-11-19 18:53:18 +01003441 ? {[val]: TreeWalk(dir .. '/' .. val)}
Bram Moolenaar08f7a412020-07-13 20:41:08 +02003442 : val
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01003443 )
Bram Moolenaar08f7a412020-07-13 20:41:08 +02003444enddef
3445
3446def Test_closure_in_map()
Bram Moolenaarf5fec052022-09-11 11:49:22 +01003447 mkdir('XclosureDir/tdir', 'pR')
Bram Moolenaar08f7a412020-07-13 20:41:08 +02003448 writefile(['111'], 'XclosureDir/file1')
3449 writefile(['222'], 'XclosureDir/file2')
3450 writefile(['333'], 'XclosureDir/tdir/file3')
3451
Bram Moolenaare0de1712020-12-02 17:36:54 +01003452 TreeWalk('XclosureDir')->assert_equal(['file1', 'file2', {tdir: ['file3']}])
Bram Moolenaar08f7a412020-07-13 20:41:08 +02003453enddef
3454
Bram Moolenaar7b5d5442020-10-04 13:42:34 +02003455def Test_invalid_function_name()
3456 var lines =<< trim END
3457 vim9script
3458 def s: list<string>
3459 END
Bram Moolenaara749a422022-02-12 19:52:25 +00003460 v9.CheckScriptFailure(lines, 'E1268:')
Bram Moolenaar7b5d5442020-10-04 13:42:34 +02003461
3462 lines =<< trim END
3463 vim9script
3464 def g: list<string>
3465 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003466 v9.CheckScriptFailure(lines, 'E129:')
Bram Moolenaar7b5d5442020-10-04 13:42:34 +02003467
3468 lines =<< trim END
3469 vim9script
3470 def <SID>: list<string>
3471 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003472 v9.CheckScriptFailure(lines, 'E884:')
Bram Moolenaar7b5d5442020-10-04 13:42:34 +02003473
3474 lines =<< trim END
3475 vim9script
3476 def F list<string>
3477 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003478 v9.CheckScriptFailure(lines, 'E488:')
Bram Moolenaar7b5d5442020-10-04 13:42:34 +02003479enddef
3480
Bram Moolenaara90afb92020-07-15 22:38:56 +02003481def Test_partial_call()
Bram Moolenaarf78da4f2021-08-01 15:40:31 +02003482 var lines =<< trim END
3483 var Xsetlist: func
3484 Xsetlist = function('setloclist', [0])
3485 Xsetlist([], ' ', {title: 'test'})
3486 getloclist(0, {title: 1})->assert_equal({title: 'test'})
Bram Moolenaara90afb92020-07-15 22:38:56 +02003487
Bram Moolenaarf78da4f2021-08-01 15:40:31 +02003488 Xsetlist = function('setloclist', [0, [], ' '])
3489 Xsetlist({title: 'test'})
3490 getloclist(0, {title: 1})->assert_equal({title: 'test'})
Bram Moolenaara90afb92020-07-15 22:38:56 +02003491
Bram Moolenaarf78da4f2021-08-01 15:40:31 +02003492 Xsetlist = function('setqflist')
3493 Xsetlist([], ' ', {title: 'test'})
3494 getqflist({title: 1})->assert_equal({title: 'test'})
Bram Moolenaara90afb92020-07-15 22:38:56 +02003495
Bram Moolenaarf78da4f2021-08-01 15:40:31 +02003496 Xsetlist = function('setqflist', [[], ' '])
3497 Xsetlist({title: 'test'})
3498 getqflist({title: 1})->assert_equal({title: 'test'})
Bram Moolenaar6abd3dc2020-10-04 14:17:32 +02003499
Bram Moolenaarf78da4f2021-08-01 15:40:31 +02003500 var Len: func: number = function('len', ['word'])
3501 assert_equal(4, Len())
3502
3503 var RepeatFunc = function('repeat', ['o'])
3504 assert_equal('ooooo', RepeatFunc(5))
3505 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003506 v9.CheckDefAndScriptSuccess(lines)
Bram Moolenaarc66f6452021-08-19 21:08:30 +02003507
3508 lines =<< trim END
3509 vim9script
3510 def Foo(Parser: any)
3511 enddef
3512 var Expr: func(dict<any>): dict<any>
3513 const Call = Foo(Expr)
3514 END
Bram Moolenaar8acb9cc2022-03-08 13:18:55 +00003515 v9.CheckScriptFailure(lines, 'E1031:')
Bram Moolenaara90afb92020-07-15 22:38:56 +02003516enddef
3517
Bram Moolenaarcd1cda22022-02-16 21:48:25 +00003518def Test_partial_double_nested()
3519 var idx = 123
3520 var Get = () => idx
3521 var Ref = function(Get, [])
3522 var RefRef = function(Ref, [])
3523 assert_equal(123, RefRef())
3524enddef
3525
Bram Moolenaar673bcb12022-03-08 16:52:24 +00003526def Test_partial_null_function()
3527 var lines =<< trim END
3528 var d: dict<func> = {f: null_function}
3529 var Ref = d.f
Bram Moolenaared0c62e2022-03-08 19:43:55 +00003530 assert_equal('func(...): unknown', typename(Ref))
Bram Moolenaar673bcb12022-03-08 16:52:24 +00003531 END
3532 v9.CheckDefAndScriptSuccess(lines)
3533enddef
3534
Bram Moolenaarfe1bfc92022-02-06 13:55:03 +00003535" Using "idx" from a legacy global function does not work.
3536" This caused a crash when called from legacy context.
3537func Test_partial_call_fails()
3538 let lines =<< trim END
3539 vim9script
3540
3541 var l = ['a', 'b', 'c']
3542 def Iter(container: any): any
3543 var idx = -1
3544 var obj = {state: container}
Bram Moolenaarf681cfb2022-02-07 20:30:57 +00003545 def g:NextItem__(self: dict<any>): any
Bram Moolenaarfe1bfc92022-02-06 13:55:03 +00003546 ++idx
3547 return self.state[idx]
3548 enddef
Bram Moolenaarf681cfb2022-02-07 20:30:57 +00003549 obj.__next__ = function('g:NextItem__', [obj])
Bram Moolenaarfe1bfc92022-02-06 13:55:03 +00003550 return obj
3551 enddef
3552
3553 var it = Iter(l)
3554 echo it.__next__()
3555 END
Bram Moolenaarf5fec052022-09-11 11:49:22 +01003556 call writefile(lines, 'XpartialCall', 'D')
Bram Moolenaarfe1bfc92022-02-06 13:55:03 +00003557 try
3558 source XpartialCall
3559 catch /E1248:/
3560 endtry
Bram Moolenaarfe1bfc92022-02-06 13:55:03 +00003561endfunc
3562
Bram Moolenaar2dd0a2c2020-08-08 15:10:27 +02003563def Test_cmd_modifier()
3564 tab echo '0'
Bram Moolenaar62aec932022-01-29 21:45:34 +00003565 v9.CheckDefFailure(['5tab echo 3'], 'E16:')
Bram Moolenaar2dd0a2c2020-08-08 15:10:27 +02003566enddef
3567
3568def Test_restore_modifiers()
3569 # check that when compiling a :def function command modifiers are not messed
3570 # up.
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02003571 var lines =<< trim END
Bram Moolenaar2dd0a2c2020-08-08 15:10:27 +02003572 vim9script
3573 set eventignore=
3574 autocmd QuickFixCmdPost * copen
3575 def AutocmdsDisabled()
Bram Moolenaarc3235272021-07-10 19:42:03 +02003576 eval 1 + 2
Bram Moolenaar2dd0a2c2020-08-08 15:10:27 +02003577 enddef
3578 func Func()
3579 noautocmd call s:AutocmdsDisabled()
3580 let g:ei_after = &eventignore
3581 endfunc
3582 Func()
3583 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003584 v9.CheckScriptSuccess(lines)
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02003585 g:ei_after->assert_equal('')
Bram Moolenaar2dd0a2c2020-08-08 15:10:27 +02003586enddef
3587
Bram Moolenaardfa3d552020-09-10 22:05:08 +02003588def StackTop()
Bram Moolenaarc3235272021-07-10 19:42:03 +02003589 eval 1 + 2
3590 eval 2 + 3
Bram Moolenaardfa3d552020-09-10 22:05:08 +02003591 # call not on fourth line
Bram Moolenaar62aec932022-01-29 21:45:34 +00003592 g:StackBot()
Bram Moolenaardfa3d552020-09-10 22:05:08 +02003593enddef
3594
3595def StackBot()
3596 # throw an error
3597 eval [][0]
3598enddef
3599
3600def Test_callstack_def()
3601 try
Bram Moolenaar62aec932022-01-29 21:45:34 +00003602 g:StackTop()
Bram Moolenaardfa3d552020-09-10 22:05:08 +02003603 catch
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02003604 v:throwpoint->assert_match('Test_callstack_def\[2\]..StackTop\[4\]..StackBot, line 2')
Bram Moolenaardfa3d552020-09-10 22:05:08 +02003605 endtry
3606enddef
3607
Bram Moolenaare8211a32020-10-09 22:04:29 +02003608" Re-using spot for variable used in block
3609def Test_block_scoped_var()
3610 var lines =<< trim END
3611 vim9script
3612 def Func()
3613 var x = ['a', 'b', 'c']
3614 if 1
3615 var y = 'x'
Bram Moolenaarbb8a7ce2021-04-10 20:10:26 +02003616 map(x, (_, _) => y)
Bram Moolenaare8211a32020-10-09 22:04:29 +02003617 endif
3618 var z = x
3619 assert_equal(['x', 'x', 'x'], z)
3620 enddef
3621 Func()
3622 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003623 v9.CheckScriptSuccess(lines)
Bram Moolenaare8211a32020-10-09 22:04:29 +02003624enddef
3625
Bram Moolenaareeece9e2020-11-20 19:26:48 +01003626def Test_reset_did_emsg()
3627 var lines =<< trim END
3628 @s = 'blah'
3629 au BufWinLeave * #
3630 def Func()
3631 var winid = popup_create('popup', {})
3632 exe '*s'
3633 popup_close(winid)
3634 enddef
3635 Func()
3636 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003637 v9.CheckScriptFailure(lines, 'E492:', 8)
Bram Moolenaar2d870f82020-12-05 13:41:01 +01003638 delfunc! g:Func
Bram Moolenaareeece9e2020-11-20 19:26:48 +01003639enddef
3640
Bram Moolenaar57f799e2020-12-12 20:42:19 +01003641def Test_did_emsg_reset()
3642 # executing an autocommand resets did_emsg, this should not result in a
3643 # builtin function considered failing
3644 var lines =<< trim END
3645 vim9script
3646 au BufWinLeave * #
3647 def Func()
Bram Moolenaar767034c2021-04-09 17:24:52 +02003648 popup_menu('', {callback: (a, b) => popup_create('', {})->popup_close()})
Bram Moolenaar57f799e2020-12-12 20:42:19 +01003649 eval [][0]
3650 enddef
3651 nno <F3> <cmd>call <sid>Func()<cr>
3652 feedkeys("\<F3>\e", 'xt')
3653 END
Bram Moolenaarf5fec052022-09-11 11:49:22 +01003654 writefile(lines, 'XemsgReset', 'D')
Bram Moolenaar57f799e2020-12-12 20:42:19 +01003655 assert_fails('so XemsgReset', ['E684:', 'E684:'], lines, 2)
Bram Moolenaarf5fec052022-09-11 11:49:22 +01003656
Bram Moolenaar57f799e2020-12-12 20:42:19 +01003657 nunmap <F3>
3658 au! BufWinLeave
3659enddef
3660
Bram Moolenaar56602ba2020-12-05 21:22:08 +01003661def Test_abort_with_silent_call()
3662 var lines =<< trim END
3663 vim9script
3664 g:result = 'none'
3665 def Func()
3666 g:result += 3
3667 g:result = 'yes'
3668 enddef
3669 # error is silenced, but function aborts on error
3670 silent! Func()
3671 assert_equal('none', g:result)
3672 unlet g:result
3673 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003674 v9.CheckScriptSuccess(lines)
Bram Moolenaar56602ba2020-12-05 21:22:08 +01003675enddef
3676
Bram Moolenaarf665e972020-12-05 19:17:16 +01003677def Test_continues_with_silent_error()
3678 var lines =<< trim END
3679 vim9script
3680 g:result = 'none'
3681 def Func()
3682 silent! g:result += 3
3683 g:result = 'yes'
3684 enddef
3685 # error is silenced, function does not abort
3686 Func()
3687 assert_equal('yes', g:result)
3688 unlet g:result
3689 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003690 v9.CheckScriptSuccess(lines)
Bram Moolenaarf665e972020-12-05 19:17:16 +01003691enddef
3692
Bram Moolenaaraf0df472020-12-02 20:51:22 +01003693def Test_abort_even_with_silent()
3694 var lines =<< trim END
3695 vim9script
3696 g:result = 'none'
3697 def Func()
3698 eval {-> ''}() .. '' .. {}['X']
3699 g:result = 'yes'
3700 enddef
Bram Moolenaarf665e972020-12-05 19:17:16 +01003701 silent! Func()
Bram Moolenaaraf0df472020-12-02 20:51:22 +01003702 assert_equal('none', g:result)
Bram Moolenaar4029cab2020-12-05 18:13:27 +01003703 unlet g:result
3704 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003705 v9.CheckScriptSuccess(lines)
Bram Moolenaar4029cab2020-12-05 18:13:27 +01003706enddef
3707
Bram Moolenaarf665e972020-12-05 19:17:16 +01003708def Test_cmdmod_silent_restored()
3709 var lines =<< trim END
3710 vim9script
3711 def Func()
3712 g:result = 'none'
3713 silent! g:result += 3
3714 g:result = 'none'
3715 g:result += 3
3716 enddef
3717 Func()
3718 END
3719 # can't use CheckScriptFailure, it ignores the :silent!
3720 var fname = 'Xdefsilent'
Bram Moolenaarf5fec052022-09-11 11:49:22 +01003721 writefile(lines, fname, 'D')
Bram Moolenaarf665e972020-12-05 19:17:16 +01003722 var caught = 'no'
3723 try
3724 exe 'source ' .. fname
3725 catch /E1030:/
3726 caught = 'yes'
3727 assert_match('Func, line 4', v:throwpoint)
3728 endtry
3729 assert_equal('yes', caught)
Bram Moolenaarf665e972020-12-05 19:17:16 +01003730enddef
3731
Bram Moolenaar2fecb532021-03-24 22:00:56 +01003732def Test_cmdmod_silent_nested()
3733 var lines =<< trim END
3734 vim9script
3735 var result = ''
3736
3737 def Error()
3738 result ..= 'Eb'
3739 eval [][0]
3740 result ..= 'Ea'
3741 enddef
3742
3743 def Crash()
3744 result ..= 'Cb'
3745 sil! Error()
3746 result ..= 'Ca'
3747 enddef
3748
3749 Crash()
3750 assert_equal('CbEbEaCa', result)
3751 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003752 v9.CheckScriptSuccess(lines)
Bram Moolenaar2fecb532021-03-24 22:00:56 +01003753enddef
3754
Bram Moolenaar4029cab2020-12-05 18:13:27 +01003755def Test_dict_member_with_silent()
3756 var lines =<< trim END
3757 vim9script
3758 g:result = 'none'
3759 var d: dict<any>
3760 def Func()
3761 try
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01003762 g:result = map([], (_, v) => ({}[v]))->join() .. d['']
Bram Moolenaar4029cab2020-12-05 18:13:27 +01003763 catch
3764 endtry
3765 enddef
3766 silent! Func()
3767 assert_equal('0', g:result)
3768 unlet g:result
Bram Moolenaaraf0df472020-12-02 20:51:22 +01003769 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003770 v9.CheckScriptSuccess(lines)
Bram Moolenaaraf0df472020-12-02 20:51:22 +01003771enddef
3772
Bram Moolenaarf9041332021-01-21 19:41:16 +01003773def Test_skip_cmds_with_silent()
3774 var lines =<< trim END
3775 vim9script
3776
3777 def Func(b: bool)
3778 Crash()
3779 enddef
3780
3781 def Crash()
3782 sil! :/not found/d _
3783 sil! :/not found/put _
3784 enddef
3785
3786 Func(true)
3787 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003788 v9.CheckScriptSuccess(lines)
Bram Moolenaarf9041332021-01-21 19:41:16 +01003789enddef
3790
Bram Moolenaar5b3d1bb2020-12-22 12:20:08 +01003791def Test_opfunc()
Bram Moolenaar848fadd2022-01-30 15:28:30 +00003792 nnoremap <F3> <cmd>set opfunc=g:Opfunc<cr>g@
Bram Moolenaar5b3d1bb2020-12-22 12:20:08 +01003793 def g:Opfunc(_: any): string
3794 setline(1, 'ASDF')
3795 return ''
3796 enddef
3797 new
3798 setline(1, 'asdf')
3799 feedkeys("\<F3>$", 'x')
3800 assert_equal('ASDF', getline(1))
3801
3802 bwipe!
3803 nunmap <F3>
3804enddef
3805
Bram Moolenaar3b309f12021-12-13 18:19:55 +00003806func Test_opfunc_error()
3807 CheckScreendump
3808 call Run_Test_opfunc_error()
3809endfunc
3810
3811def Run_Test_opfunc_error()
3812 # test that the error from Opfunc() is displayed right away
3813 var lines =<< trim END
3814 vim9script
3815
3816 def Opfunc(type: string)
3817 try
3818 eval [][0]
3819 catch /nothing/ # error not caught
3820 endtry
3821 enddef
3822 &operatorfunc = Opfunc
3823 nnoremap <expr> l <SID>L()
3824 def L(): string
3825 return 'l'
3826 enddef
3827 'x'->repeat(10)->setline(1)
3828 feedkeys('g@l', 'n')
3829 feedkeys('llll')
3830 END
Bram Moolenaarf5fec052022-09-11 11:49:22 +01003831 call writefile(lines, 'XTest_opfunc_error', 'D')
Bram Moolenaar3b309f12021-12-13 18:19:55 +00003832
Bram Moolenaar62aec932022-01-29 21:45:34 +00003833 var buf = g:RunVimInTerminal('-S XTest_opfunc_error', {rows: 6, wait_for_ruler: 0})
3834 g:WaitForAssert(() => assert_match('Press ENTER', term_getline(buf, 6)))
Bram Moolenaarec892232022-05-06 17:53:06 +01003835 g:WaitForAssert(() => assert_match('E684: List index out of range: 0', term_getline(buf, 5)))
Bram Moolenaar3b309f12021-12-13 18:19:55 +00003836
3837 # clean up
Bram Moolenaar62aec932022-01-29 21:45:34 +00003838 g:StopVimInTerminal(buf)
Bram Moolenaar3b309f12021-12-13 18:19:55 +00003839enddef
3840
Bram Moolenaar077a4232020-12-22 18:33:27 +01003841" this was crashing on exit
3842def Test_nested_lambda_in_closure()
3843 var lines =<< trim END
3844 vim9script
Bram Moolenaar227c58a2021-04-28 20:40:44 +02003845 command WriteDone writefile(['Done'], 'XnestedDone')
Bram Moolenaar077a4232020-12-22 18:33:27 +01003846 def Outer()
3847 def g:Inner()
3848 echo map([1, 2, 3], {_, v -> v + 1})
3849 enddef
3850 g:Inner()
3851 enddef
3852 defcompile
Bram Moolenaar227c58a2021-04-28 20:40:44 +02003853 # not reached
Bram Moolenaar077a4232020-12-22 18:33:27 +01003854 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003855 if !g:RunVim([], lines, '--clean -c WriteDone -c quit')
Bram Moolenaar077a4232020-12-22 18:33:27 +01003856 return
3857 endif
3858 assert_equal(['Done'], readfile('XnestedDone'))
3859 delete('XnestedDone')
3860enddef
3861
Bram Moolenaar92368aa2022-02-07 17:50:39 +00003862def Test_nested_closure_funcref()
3863 var lines =<< trim END
3864 vim9script
3865 def Func()
3866 var n: number
3867 def Nested()
3868 ++n
3869 enddef
3870 Nested()
3871 g:result_one = n
3872 var Ref = function(Nested)
3873 Ref()
3874 g:result_two = n
3875 enddef
3876 Func()
3877 END
3878 v9.CheckScriptSuccess(lines)
3879 assert_equal(1, g:result_one)
3880 assert_equal(2, g:result_two)
3881 unlet g:result_one g:result_two
3882enddef
3883
Bram Moolenaar7aca5ca2022-02-07 19:56:43 +00003884def Test_nested_closure_in_dict()
3885 var lines =<< trim END
3886 vim9script
3887 def Func(): dict<any>
3888 var n: number
3889 def Inc(): number
3890 ++n
3891 return n
3892 enddef
3893 return {inc: function(Inc)}
3894 enddef
3895 disas Func
3896 var d = Func()
3897 assert_equal(1, d.inc())
3898 assert_equal(2, d.inc())
3899 END
3900 v9.CheckScriptSuccess(lines)
3901enddef
3902
Bram Moolenaarfb43cfc2022-03-11 18:54:17 +00003903def Test_script_local_other_script()
3904 var lines =<< trim END
3905 function LegacyJob()
3906 let FuncRef = function('s:close_cb')
3907 endfunction
3908 function s:close_cb(...)
3909 endfunction
3910 END
Bram Moolenaarf5fec052022-09-11 11:49:22 +01003911 lines->writefile('Xlegacy.vim', 'D')
Bram Moolenaarfb43cfc2022-03-11 18:54:17 +00003912 source Xlegacy.vim
3913 g:LegacyJob()
3914 g:LegacyJob()
3915 g:LegacyJob()
3916
3917 delfunc g:LegacyJob
Bram Moolenaarfb43cfc2022-03-11 18:54:17 +00003918enddef
3919
Bram Moolenaar04947cc2021-03-06 19:26:46 +01003920def Test_check_func_arg_types()
3921 var lines =<< trim END
3922 vim9script
3923 def F1(x: string): string
3924 return x
3925 enddef
3926
3927 def F2(x: number): number
3928 return x + 1
3929 enddef
3930
Bram Moolenaar1d9cef72022-03-17 16:30:03 +00003931 def G(Fg: func): dict<func>
3932 return {f: Fg}
Bram Moolenaar04947cc2021-03-06 19:26:46 +01003933 enddef
3934
3935 def H(d: dict<func>): string
3936 return d.f('a')
3937 enddef
3938 END
3939
Bram Moolenaar62aec932022-01-29 21:45:34 +00003940 v9.CheckScriptSuccess(lines + ['echo H(G(F1))'])
3941 v9.CheckScriptFailure(lines + ['echo H(G(F2))'], 'E1013:')
Bram Moolenaar1d9cef72022-03-17 16:30:03 +00003942
3943 v9.CheckScriptFailure(lines + ['def SomeFunc(ff: func)', 'enddef'], 'E704:')
Bram Moolenaar04947cc2021-03-06 19:26:46 +01003944enddef
3945
Bram Moolenaarbadf04f2022-03-12 21:28:22 +00003946def Test_call_func_with_null()
3947 var lines =<< trim END
3948 def Fstring(v: string)
3949 assert_equal(null_string, v)
3950 enddef
3951 Fstring(null_string)
3952 def Fblob(v: blob)
3953 assert_equal(null_blob, v)
3954 enddef
3955 Fblob(null_blob)
3956 def Flist(v: list<number>)
3957 assert_equal(null_list, v)
3958 enddef
3959 Flist(null_list)
3960 def Fdict(v: dict<number>)
3961 assert_equal(null_dict, v)
3962 enddef
3963 Fdict(null_dict)
3964 def Ffunc(Fv: func(number): number)
3965 assert_equal(null_function, Fv)
3966 enddef
3967 Ffunc(null_function)
3968 if has('channel')
3969 def Fchannel(v: channel)
3970 assert_equal(null_channel, v)
3971 enddef
3972 Fchannel(null_channel)
3973 def Fjob(v: job)
3974 assert_equal(null_job, v)
3975 enddef
3976 Fjob(null_job)
3977 endif
3978 END
3979 v9.CheckDefAndScriptSuccess(lines)
3980enddef
3981
3982def Test_null_default_argument()
3983 var lines =<< trim END
3984 def Fstring(v: string = null_string)
3985 assert_equal(null_string, v)
3986 enddef
3987 Fstring()
3988 def Fblob(v: blob = null_blob)
3989 assert_equal(null_blob, v)
3990 enddef
3991 Fblob()
3992 def Flist(v: list<number> = null_list)
3993 assert_equal(null_list, v)
3994 enddef
3995 Flist()
3996 def Fdict(v: dict<number> = null_dict)
3997 assert_equal(null_dict, v)
3998 enddef
3999 Fdict()
4000 def Ffunc(Fv: func(number): number = null_function)
4001 assert_equal(null_function, Fv)
4002 enddef
4003 Ffunc()
4004 if has('channel')
4005 def Fchannel(v: channel = null_channel)
4006 assert_equal(null_channel, v)
4007 enddef
4008 Fchannel()
4009 def Fjob(v: job = null_job)
4010 assert_equal(null_job, v)
4011 enddef
4012 Fjob()
4013 endif
4014 END
4015 v9.CheckDefAndScriptSuccess(lines)
4016enddef
4017
4018def Test_null_return()
4019 var lines =<< trim END
4020 def Fstring(): string
4021 return null_string
4022 enddef
4023 assert_equal(null_string, Fstring())
4024 def Fblob(): blob
4025 return null_blob
4026 enddef
4027 assert_equal(null_blob, Fblob())
4028 def Flist(): list<number>
4029 return null_list
4030 enddef
4031 assert_equal(null_list, Flist())
4032 def Fdict(): dict<number>
4033 return null_dict
4034 enddef
4035 assert_equal(null_dict, Fdict())
4036 def Ffunc(): func(number): number
4037 return null_function
4038 enddef
4039 assert_equal(null_function, Ffunc())
4040 if has('channel')
4041 def Fchannel(): channel
4042 return null_channel
4043 enddef
4044 assert_equal(null_channel, Fchannel())
4045 def Fjob(): job
4046 return null_job
4047 enddef
4048 assert_equal(null_job, Fjob())
4049 endif
4050 END
4051 v9.CheckDefAndScriptSuccess(lines)
4052enddef
4053
Bram Moolenaar6e48b842021-08-10 22:52:02 +02004054def Test_list_any_type_checked()
4055 var lines =<< trim END
4056 vim9script
4057 def Foo()
4058 --decl--
4059 Bar(l)
4060 enddef
4061 def Bar(ll: list<dict<any>>)
4062 enddef
4063 Foo()
4064 END
Bram Moolenaar2d3ac2e2022-02-03 12:34:05 +00004065 # "any" could be "dict<any>", thus OK
Bram Moolenaar6e48b842021-08-10 22:52:02 +02004066 lines[2] = 'var l: list<any>'
Bram Moolenaar2d3ac2e2022-02-03 12:34:05 +00004067 v9.CheckScriptSuccess(lines)
Bram Moolenaar6e48b842021-08-10 22:52:02 +02004068 lines[2] = 'var l: list<any> = []'
Bram Moolenaar2d3ac2e2022-02-03 12:34:05 +00004069 v9.CheckScriptSuccess(lines)
Bram Moolenaar6e48b842021-08-10 22:52:02 +02004070
4071 lines[2] = 'var l: list<any> = [11]'
Bram Moolenaar62aec932022-01-29 21:45:34 +00004072 v9.CheckScriptFailure(lines, 'E1013: Argument 1: type mismatch, expected list<dict<any>> but got list<number>', 2)
Bram Moolenaar6e48b842021-08-10 22:52:02 +02004073enddef
4074
Bram Moolenaar701cc6c2021-04-10 13:33:48 +02004075def Test_compile_error()
4076 var lines =<< trim END
4077 def g:Broken()
4078 echo 'a' + {}
4079 enddef
4080 call g:Broken()
4081 END
4082 # First call: compilation error
Bram Moolenaar62aec932022-01-29 21:45:34 +00004083 v9.CheckScriptFailure(lines, 'E1051: Wrong argument type for +')
Bram Moolenaar701cc6c2021-04-10 13:33:48 +02004084
4085 # Second call won't try compiling again
4086 assert_fails('call g:Broken()', 'E1091: Function is not compiled: Broken')
Bram Moolenaar599410c2021-04-10 14:03:43 +02004087 delfunc g:Broken
4088
4089 # No error when compiling with :silent!
4090 lines =<< trim END
4091 def g:Broken()
4092 echo 'a' + []
4093 enddef
4094 silent! defcompile
4095 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00004096 v9.CheckScriptSuccess(lines)
Bram Moolenaar599410c2021-04-10 14:03:43 +02004097
4098 # Calling the function won't try compiling again
4099 assert_fails('call g:Broken()', 'E1091: Function is not compiled: Broken')
4100 delfunc g:Broken
Bram Moolenaar701cc6c2021-04-10 13:33:48 +02004101enddef
4102
Bram Moolenaar962c43b2021-04-10 17:18:09 +02004103def Test_ignored_argument()
4104 var lines =<< trim END
4105 vim9script
4106 def Ignore(_, _): string
4107 return 'yes'
4108 enddef
4109 assert_equal('yes', Ignore(1, 2))
4110
4111 func Ok(_)
4112 return a:_
4113 endfunc
4114 assert_equal('ok', Ok('ok'))
4115
4116 func Oktoo()
4117 let _ = 'too'
4118 return _
4119 endfunc
4120 assert_equal('too', Oktoo())
Bram Moolenaarda479c72021-04-10 21:01:38 +02004121
4122 assert_equal([[1], [2], [3]], range(3)->mapnew((_, v) => [v]->map((_, w) => w + 1)))
Bram Moolenaar962c43b2021-04-10 17:18:09 +02004123 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00004124 v9.CheckScriptSuccess(lines)
Bram Moolenaar962c43b2021-04-10 17:18:09 +02004125
4126 lines =<< trim END
4127 def Ignore(_: string): string
4128 return _
4129 enddef
4130 defcompile
4131 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00004132 v9.CheckScriptFailure(lines, 'E1181:', 1)
Bram Moolenaar962c43b2021-04-10 17:18:09 +02004133
4134 lines =<< trim END
4135 var _ = 1
4136 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00004137 v9.CheckDefAndScriptFailure(lines, 'E1181:', 1)
Yegappan Lakshmanan34fcb692021-05-25 20:14:00 +02004138
4139 lines =<< trim END
4140 var x = _
4141 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00004142 v9.CheckDefAndScriptFailure(lines, 'E1181:', 1)
Bram Moolenaar962c43b2021-04-10 17:18:09 +02004143enddef
4144
Bram Moolenaarbb8a7ce2021-04-10 20:10:26 +02004145def Test_too_many_arguments()
4146 var lines =<< trim END
4147 echo [0, 1, 2]->map(() => 123)
4148 END
Bram Moolenaareddd4fc2022-02-20 15:52:28 +00004149 v9.CheckDefAndScriptFailure(lines, ['E176:', 'E1106: 2 arguments too many'], 1)
Bram Moolenaarbb8a7ce2021-04-10 20:10:26 +02004150
4151 lines =<< trim END
4152 echo [0, 1, 2]->map((_) => 123)
4153 END
Bram Moolenaareddd4fc2022-02-20 15:52:28 +00004154 v9.CheckDefAndScriptFailure(lines, ['E176', 'E1106: One argument too many'], 1)
Bram Moolenaar31d99482022-05-26 22:24:43 +01004155
4156 lines =<< trim END
4157 vim9script
4158 def OneArgument(arg: string)
4159 echo arg
4160 enddef
4161 var Ref = OneArgument
4162 Ref('a', 'b')
4163 END
4164 v9.CheckScriptFailure(lines, 'E118:')
4165enddef
4166
4167def Test_funcref_with_base()
4168 var lines =<< trim END
4169 vim9script
4170 def TwoArguments(str: string, nr: number)
4171 echo str nr
4172 enddef
4173 var Ref = TwoArguments
4174 Ref('a', 12)
4175 'b'->Ref(34)
4176 END
4177 v9.CheckScriptSuccess(lines)
4178
4179 lines =<< trim END
4180 vim9script
4181 def TwoArguments(str: string, nr: number)
4182 echo str nr
4183 enddef
4184 var Ref = TwoArguments
4185 'a'->Ref('b')
4186 END
4187 v9.CheckScriptFailure(lines, 'E1013: Argument 2: type mismatch, expected number but got string', 6)
4188
4189 lines =<< trim END
4190 vim9script
4191 def TwoArguments(str: string, nr: number)
4192 echo str nr
4193 enddef
4194 var Ref = TwoArguments
4195 123->Ref(456)
4196 END
4197 v9.CheckScriptFailure(lines, 'E1013: Argument 1: type mismatch, expected string but got number')
4198
4199 lines =<< trim END
4200 vim9script
4201 def TwoArguments(nr: number, str: string)
4202 echo str nr
4203 enddef
4204 var Ref = TwoArguments
4205 123->Ref('b')
4206 def AndNowCompiled()
4207 456->Ref('x')
4208 enddef
4209 AndNowCompiled()
4210 END
4211 v9.CheckScriptSuccess(lines)
Bram Moolenaarbb8a7ce2021-04-10 20:10:26 +02004212enddef
Bram Moolenaar077a4232020-12-22 18:33:27 +01004213
Bram Moolenaara6aa1642021-04-23 19:32:23 +02004214def Test_closing_brace_at_start_of_line()
4215 var lines =<< trim END
4216 def Func()
4217 enddef
4218 Func(
4219 )
4220 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00004221 v9.CheckDefAndScriptSuccess(lines)
Bram Moolenaara6aa1642021-04-23 19:32:23 +02004222enddef
4223
Bram Moolenaar62aec932022-01-29 21:45:34 +00004224func s:CreateMydict()
Bram Moolenaarb033ee22021-08-15 16:08:36 +02004225 let g:mydict = {}
4226 func g:mydict.afunc()
4227 let g:result = self.key
4228 endfunc
4229endfunc
4230
4231def Test_numbered_function_reference()
4232 CreateMydict()
4233 var output = execute('legacy func g:mydict.afunc')
4234 var funcName = 'g:' .. substitute(output, '.*function \(\d\+\).*', '\1', '')
4235 execute 'function(' .. funcName .. ', [], {key: 42})()'
4236 # check that the function still exists
4237 assert_equal(output, execute('legacy func g:mydict.afunc'))
4238 unlet g:mydict
4239enddef
4240
Bram Moolenaard3a11782022-01-05 16:50:40 +00004241def Test_go_beyond_end_of_cmd()
4242 # this was reading the byte after the end of the line
4243 var lines =<< trim END
4244 def F()
4245 cal
4246 enddef
4247 defcompile
4248 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00004249 v9.CheckScriptFailure(lines, 'E476:')
Bram Moolenaard3a11782022-01-05 16:50:40 +00004250enddef
4251
Yegappan Lakshmanan7c7e19c2022-04-09 11:09:07 +01004252" Test for memory allocation failure when defining a new lambda
4253func Test_lambda_allocation_failure()
4254 new
4255 let lines =<< trim END
4256 vim9script
4257 g:Xlambda = (x): number => {
4258 return x + 1
4259 }
4260 END
4261 call setline(1, lines)
4262 call test_alloc_fail(GetAllocId('get_func'), 0, 0)
4263 call assert_fails('source', 'E342:')
4264 call assert_false(exists('g:Xlambda'))
4265 bw!
4266endfunc
4267
Bram Moolenaarbce69d62022-05-22 13:45:52 +01004268def Test_multiple_funcref()
4269 # This was using a NULL pointer
4270 var lines =<< trim END
4271 vim9script
4272 def A(F: func, ...args: list<any>): func
4273 return funcref(F, args)
4274 enddef
4275
4276 def B(F: func): func
4277 return funcref(A, [F])
4278 enddef
4279
4280 def Test(n: number)
4281 enddef
4282
4283 const X = B(Test)
4284 X(1)
4285 END
4286 v9.CheckScriptSuccess(lines)
4287
4288 # slightly different case
4289 lines =<< trim END
4290 vim9script
4291
4292 def A(F: func, ...args: list<any>): any
4293 return call(F, args)
4294 enddef
4295
4296 def B(F: func): func
4297 return funcref(A, [F])
4298 enddef
4299
4300 def Test(n: number)
4301 enddef
4302
4303 const X = B(Test)
4304 X(1)
4305 END
4306 v9.CheckScriptSuccess(lines)
4307enddef
4308
Bram Moolenaarbd683e32022-07-18 17:49:03 +01004309def Test_cexpr_errmsg_line_number()
4310 var lines =<< trim END
4311 vim9script
4312 def Func()
4313 var qfl = {}
4314 cexpr qfl
4315 enddef
4316 Func()
4317 END
4318 v9.CheckScriptFailure(lines, 'E777', 2)
4319enddef
4320
Bram Moolenaar1d84f762022-09-03 21:35:53 +01004321def AddDefer(s: string)
4322 g:deferred->extend([s])
4323enddef
4324
4325def DeferTwo()
4326 g:deferred->extend(['in Two'])
4327 for n in range(3)
4328 defer g:AddDefer('two' .. n)
4329 endfor
4330 g:deferred->extend(['end Two'])
4331enddef
4332
4333def DeferOne()
4334 g:deferred->extend(['in One'])
4335 defer g:AddDefer('one')
4336 g:DeferTwo()
4337 g:deferred->extend(['end One'])
4338
4339 writefile(['text'], 'XdeferFile')
4340 defer delete('XdeferFile')
4341enddef
4342
4343def Test_defer()
4344 g:deferred = []
4345 g:DeferOne()
4346 assert_equal(['in One', 'in Two', 'end Two', 'two2', 'two1', 'two0', 'end One', 'one'], g:deferred)
4347 unlet g:deferred
4348 assert_equal('', glob('XdeferFile'))
4349enddef
4350
Bram Moolenaar8b716f52022-02-15 21:17:56 +00004351" The following messes up syntax highlight, keep near the end.
Bram Moolenaar20677332021-06-06 17:02:53 +02004352if has('python3')
Bram Moolenaar8b716f52022-02-15 21:17:56 +00004353 def Test_python3_command()
4354 py3 import vim
Bram Moolenaarf5288c52022-02-15 21:33:29 +00004355 py3 vim.command("g:done = 'yes'")
Bram Moolenaar8b716f52022-02-15 21:17:56 +00004356 assert_equal('yes', g:done)
4357 unlet g:done
4358 enddef
4359
Bram Moolenaar20677332021-06-06 17:02:53 +02004360 def Test_python3_heredoc()
4361 py3 << trim EOF
4362 import vim
4363 vim.vars['didit'] = 'yes'
4364 EOF
4365 assert_equal('yes', g:didit)
4366
4367 python3 << trim EOF
4368 import vim
4369 vim.vars['didit'] = 'again'
4370 EOF
4371 assert_equal('again', g:didit)
4372 enddef
4373endif
4374
Bram Moolenaar20677332021-06-06 17:02:53 +02004375if has('lua')
4376 def Test_lua_heredoc()
4377 g:d = {}
4378 lua << trim EOF
4379 x = vim.eval('g:d')
4380 x['key'] = 'val'
4381 EOF
4382 assert_equal('val', g:d.key)
4383 enddef
Bram Moolenaarefd73ae2022-03-20 18:51:00 +00004384
4385 def Test_lua_heredoc_fails()
4386 var lines = [
4387 'vim9script',
4388 'def ExeLua()',
4389 'lua << trim EOLUA',
4390 "x = vim.eval('g:nodict')",
4391 'EOLUA',
4392 'enddef',
4393 'ExeLua()',
4394 ]
4395 v9.CheckScriptFailure(lines, 'E121: Undefined variable: g:nodict')
4396 enddef
Bram Moolenaar20677332021-06-06 17:02:53 +02004397endif
4398
Bram Moolenaard881d152022-05-13 13:50:36 +01004399if has('perl')
4400 def Test_perl_heredoc_nested()
4401 var lines =<< trim END
4402 vim9script
4403 def F(): string
4404 def G(): string
4405 perl << EOF
4406 EOF
4407 return 'done'
4408 enddef
4409 return G()
4410 enddef
4411 assert_equal('done', F())
4412 END
4413 v9.CheckScriptSuccess(lines)
4414 enddef
4415endif
4416
Bram Moolenaarf7779c62020-05-03 15:38:16 +02004417
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02004418" vim: ts=8 sw=2 sts=2 expandtab tw=80 fdm=marker