blob: e68cbcf152d68c35db4d21ec7e334fc1cbc50991 [file] [log] [blame]
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001" Test various aspects of the Vim9 script language.
2
3source check.vim
Bram Moolenaarad304702020-09-06 18:22:53 +02004source term_util.vim
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02005source view_util.vim
Bram Moolenaar62aec932022-01-29 21:45:34 +00006import './vim9.vim' as v9
Bram Moolenaar47e7d702020-07-05 18:18:42 +02007source screendump.vim
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02008
9func Test_def_basic()
10 def SomeFunc(): string
11 return 'yes'
12 enddef
Bram Moolenaarc0c71e92020-09-11 19:09:48 +020013 call SomeFunc()->assert_equal('yes')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +020014endfunc
15
Bram Moolenaar2b9b17e2020-10-13 18:38:11 +020016func Test_compiling_error()
17 " use a terminal to see the whole error message
Bram Moolenaarf4e8cdd2020-10-12 22:07:13 +020018 CheckRunVimInTerminal
19
Bram Moolenaar2b9b17e2020-10-13 18:38:11 +020020 call TestCompilingError()
Bram Moolenaare8c46602021-04-05 22:27:37 +020021 call TestCompilingErrorInTry()
Bram Moolenaar2b9b17e2020-10-13 18:38:11 +020022endfunc
23
24def TestCompilingError()
Bram Moolenaarf4e8cdd2020-10-12 22:07:13 +020025 var lines =<< trim END
26 vim9script
27 def Fails()
28 echo nothing
29 enddef
30 defcompile
31 END
Bram Moolenaare8c46602021-04-05 22:27:37 +020032 writefile(lines, 'XTest_compile_error')
Bram Moolenaar62aec932022-01-29 21:45:34 +000033 var buf = g:RunVimInTerminal('-S XTest_compile_error',
Bram Moolenaare0de1712020-12-02 17:36:54 +010034 {rows: 10, wait_for_ruler: 0})
Bram Moolenaar62aec932022-01-29 21:45:34 +000035 g:WaitForAssert(() => assert_match('Error detected while compiling command line.*Fails.*Variable not found: nothing',
36 g:Term_getlines(buf, range(1, 9))))
Bram Moolenaarf4e8cdd2020-10-12 22:07:13 +020037
38 # clean up
Bram Moolenaar62aec932022-01-29 21:45:34 +000039 g:StopVimInTerminal(buf)
Bram Moolenaare8c46602021-04-05 22:27:37 +020040 delete('XTest_compile_error')
41enddef
42
43def TestCompilingErrorInTry()
44 var dir = 'Xdir/autoload'
45 mkdir(dir, 'p')
46
47 var lines =<< trim END
48 vim9script
Bram Moolenaard8fe6d32022-01-30 18:40:44 +000049 export def OnlyCompiled()
Bram Moolenaare8c46602021-04-05 22:27:37 +020050 g:runtime = 'yes'
51 invalid
52 enddef
53 END
54 writefile(lines, dir .. '/script.vim')
55
56 lines =<< trim END
57 vim9script
58 todo
59 try
60 script#OnlyCompiled()
61 catch /nothing/
62 endtry
63 END
64 lines[1] = 'set rtp=' .. getcwd() .. '/Xdir'
65 writefile(lines, 'XTest_compile_error')
66
Bram Moolenaar62aec932022-01-29 21:45:34 +000067 var buf = g:RunVimInTerminal('-S XTest_compile_error', {rows: 10, wait_for_ruler: 0})
68 g:WaitForAssert(() => assert_match('Error detected while compiling command line.*function script#OnlyCompiled.*Invalid command: invalid',
69 g:Term_getlines(buf, range(1, 9))))
Bram Moolenaare8c46602021-04-05 22:27:37 +020070
71 # clean up
Bram Moolenaar62aec932022-01-29 21:45:34 +000072 g:StopVimInTerminal(buf)
Bram Moolenaare8c46602021-04-05 22:27:37 +020073 delete('XTest_compile_error')
74 delete('Xdir', 'rf')
Bram Moolenaarf4e8cdd2020-10-12 22:07:13 +020075enddef
76
Bram Moolenaarb55d6182021-06-08 22:01:53 +020077def Test_compile_error_in_called_function()
78 var lines =<< trim END
79 vim9script
80 var n: number
81 def Foo()
82 &hls = n
83 enddef
84 def Bar()
85 Foo()
86 enddef
87 silent! Foo()
88 Bar()
89 END
Bram Moolenaar62aec932022-01-29 21:45:34 +000090 v9.CheckScriptFailureList(lines, ['E1012:', 'E1191:'])
Bram Moolenaarb55d6182021-06-08 22:01:53 +020091enddef
92
Bram Moolenaar22f17a22021-06-21 20:48:58 +020093def Test_wrong_function_name()
94 var lines =<< trim END
95 vim9script
96 func _Foo()
97 echo 'foo'
98 endfunc
99 END
Bram Moolenaar3787f262022-02-07 21:54:01 +0000100 v9.CheckScriptFailure(lines, 'E1267:')
Bram Moolenaar22f17a22021-06-21 20:48:58 +0200101
102 lines =<< trim END
103 vim9script
104 def _Foo()
105 echo 'foo'
106 enddef
107 END
Bram Moolenaar3787f262022-02-07 21:54:01 +0000108 v9.CheckScriptFailure(lines, 'E1267:')
Bram Moolenaar22f17a22021-06-21 20:48:58 +0200109enddef
110
Bram Moolenaarf48b2fa2021-04-12 22:02:36 +0200111def Test_autoload_name_mismatch()
112 var dir = 'Xdir/autoload'
113 mkdir(dir, 'p')
114
115 var lines =<< trim END
116 vim9script
Bram Moolenaard8fe6d32022-01-30 18:40:44 +0000117 export def NoFunction()
Bram Moolenaarf48b2fa2021-04-12 22:02:36 +0200118 # comment
119 g:runtime = 'yes'
120 enddef
121 END
122 writefile(lines, dir .. '/script.vim')
123
124 var save_rtp = &rtp
125 exe 'set rtp=' .. getcwd() .. '/Xdir'
126 lines =<< trim END
127 call script#Function()
128 END
Bram Moolenaard8fe6d32022-01-30 18:40:44 +0000129 v9.CheckScriptFailure(lines, 'E117:', 1)
Bram Moolenaarf48b2fa2021-04-12 22:02:36 +0200130
131 &rtp = save_rtp
132 delete(dir, 'rf')
133enddef
134
Bram Moolenaarf0a40692021-06-11 22:05:47 +0200135def Test_autoload_names()
136 var dir = 'Xdir/autoload'
137 mkdir(dir, 'p')
138
139 var lines =<< trim END
140 func foobar#function()
141 return 'yes'
142 endfunc
143 let foobar#var = 'no'
144 END
145 writefile(lines, dir .. '/foobar.vim')
146
147 var save_rtp = &rtp
148 exe 'set rtp=' .. getcwd() .. '/Xdir'
149
150 lines =<< trim END
151 assert_equal('yes', foobar#function())
152 var Function = foobar#function
153 assert_equal('yes', Function())
154
155 assert_equal('no', foobar#var)
156 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000157 v9.CheckDefAndScriptSuccess(lines)
Bram Moolenaarf0a40692021-06-11 22:05:47 +0200158
159 &rtp = save_rtp
160 delete(dir, 'rf')
161enddef
162
Bram Moolenaar88c89c72021-08-14 14:01:05 +0200163def Test_autoload_error_in_script()
164 var dir = 'Xdir/autoload'
165 mkdir(dir, 'p')
166
167 var lines =<< trim END
168 func scripterror#function()
169 let g:called_function = 'yes'
170 endfunc
171 let 0 = 1
172 END
173 writefile(lines, dir .. '/scripterror.vim')
174
175 var save_rtp = &rtp
176 exe 'set rtp=' .. getcwd() .. '/Xdir'
177
178 g:called_function = 'no'
179 # The error in the autoload script cannot be checked with assert_fails(), use
180 # CheckDefSuccess() instead of CheckDefFailure()
181 try
Bram Moolenaar62aec932022-01-29 21:45:34 +0000182 v9.CheckDefSuccess(['scripterror#function()'])
Bram Moolenaar88c89c72021-08-14 14:01:05 +0200183 catch
184 assert_match('E121: Undefined variable: 0', v:exception)
185 endtry
186 assert_equal('no', g:called_function)
187
188 lines =<< trim END
189 func scriptcaught#function()
190 let g:called_function = 'yes'
191 endfunc
192 try
193 let 0 = 1
194 catch
195 let g:caught = v:exception
196 endtry
197 END
198 writefile(lines, dir .. '/scriptcaught.vim')
199
200 g:called_function = 'no'
Bram Moolenaar62aec932022-01-29 21:45:34 +0000201 v9.CheckDefSuccess(['scriptcaught#function()'])
Bram Moolenaar88c89c72021-08-14 14:01:05 +0200202 assert_match('E121: Undefined variable: 0', g:caught)
203 assert_equal('yes', g:called_function)
204
205 &rtp = save_rtp
206 delete(dir, 'rf')
207enddef
208
Bram Moolenaar62aec932022-01-29 21:45:34 +0000209def s:CallRecursive(n: number): number
Bram Moolenaar0ba48e82020-11-17 18:23:19 +0100210 return CallRecursive(n + 1)
211enddef
212
Bram Moolenaar62aec932022-01-29 21:45:34 +0000213def s:CallMapRecursive(l: list<number>): number
Bram Moolenaar2949cfd2020-12-31 21:28:47 +0100214 return map(l, (_, v) => CallMapRecursive([v]))[0]
Bram Moolenaar0ba48e82020-11-17 18:23:19 +0100215enddef
216
217def Test_funcdepth_error()
218 set maxfuncdepth=10
219
220 var caught = false
221 try
222 CallRecursive(1)
223 catch /E132:/
224 caught = true
225 endtry
226 assert_true(caught)
227
228 caught = false
229 try
230 CallMapRecursive([1])
231 catch /E132:/
232 caught = true
233 endtry
234 assert_true(caught)
235
236 set maxfuncdepth&
237enddef
238
Bram Moolenaar5178b1b2021-01-01 18:43:51 +0100239def Test_endfunc_enddef()
240 var lines =<< trim END
241 def Test()
242 echo 'test'
243 endfunc
244 enddef
245 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000246 v9.CheckScriptFailure(lines, 'E1151:', 3)
Bram Moolenaar5178b1b2021-01-01 18:43:51 +0100247
248 lines =<< trim END
249 def Test()
250 func Nested()
251 echo 'test'
252 enddef
253 enddef
254 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000255 v9.CheckScriptFailure(lines, 'E1152:', 4)
Bram Moolenaar49f1e9e2021-03-22 20:49:02 +0100256
257 lines =<< trim END
258 def Ok()
259 echo 'hello'
260 enddef | echo 'there'
261 def Bad()
262 echo 'hello'
263 enddef there
264 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000265 v9.CheckScriptFailure(lines, 'E1173: Text found after enddef: there', 6)
Bram Moolenaar5178b1b2021-01-01 18:43:51 +0100266enddef
267
Bram Moolenaarb8ba9b92021-01-01 18:54:34 +0100268def Test_missing_endfunc_enddef()
269 var lines =<< trim END
270 vim9script
271 def Test()
272 echo 'test'
273 endef
274 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000275 v9.CheckScriptFailure(lines, 'E1057:', 2)
Bram Moolenaarb8ba9b92021-01-01 18:54:34 +0100276
277 lines =<< trim END
278 vim9script
279 func Some()
280 echo 'test'
281 enfffunc
282 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000283 v9.CheckScriptFailure(lines, 'E126:', 2)
Bram Moolenaarb8ba9b92021-01-01 18:54:34 +0100284enddef
285
Bram Moolenaar4efd9942021-01-24 21:14:20 +0100286def Test_white_space_before_paren()
287 var lines =<< trim END
288 vim9script
289 def Test ()
290 echo 'test'
291 enddef
292 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000293 v9.CheckScriptFailure(lines, 'E1068:', 2)
Bram Moolenaar4efd9942021-01-24 21:14:20 +0100294
295 lines =<< trim END
296 vim9script
297 func Test ()
298 echo 'test'
299 endfunc
300 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000301 v9.CheckScriptFailure(lines, 'E1068:', 2)
Bram Moolenaar4efd9942021-01-24 21:14:20 +0100302
303 lines =<< trim END
304 def Test ()
305 echo 'test'
306 enddef
307 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000308 v9.CheckScriptFailure(lines, 'E1068:', 1)
Bram Moolenaar4efd9942021-01-24 21:14:20 +0100309
310 lines =<< trim END
311 func Test ()
312 echo 'test'
313 endfunc
314 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000315 v9.CheckScriptSuccess(lines)
Bram Moolenaar4efd9942021-01-24 21:14:20 +0100316enddef
317
Bram Moolenaar832ea892021-01-08 21:55:26 +0100318def Test_enddef_dict_key()
319 var d = {
320 enddef: 'x',
321 endfunc: 'y',
322 }
323 assert_equal({enddef: 'x', endfunc: 'y'}, d)
324enddef
325
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200326def ReturnString(): string
327 return 'string'
328enddef
329
330def ReturnNumber(): number
331 return 123
332enddef
333
334let g:notNumber = 'string'
335
336def ReturnGlobal(): number
337 return g:notNumber
338enddef
339
340def Test_return_something()
Bram Moolenaar62aec932022-01-29 21:45:34 +0000341 g:ReturnString()->assert_equal('string')
342 g:ReturnNumber()->assert_equal(123)
Bram Moolenaar848fadd2022-01-30 15:28:30 +0000343 assert_fails('g:ReturnGlobal()', 'E1012: Type mismatch; expected number but got string', '', 1, 'ReturnGlobal')
Bram Moolenaaref7aadb2022-01-18 18:46:07 +0000344
345 var lines =<< trim END
346 vim9script
347
348 def Msg()
349 echomsg 'in Msg()...'
350 enddef
351
352 def Func()
353 return Msg()
354 enddef
355 defcompile
356 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000357 v9.CheckScriptFailure(lines, 'E1096:')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200358enddef
359
Bram Moolenaare32e5162021-01-21 20:21:29 +0100360def Test_check_argument_type()
361 var lines =<< trim END
362 vim9script
363 def Val(a: number, b: number): number
364 return 0
365 enddef
366 def Func()
367 var x: any = true
368 Val(0, x)
369 enddef
370 disass Func
371 Func()
372 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000373 v9.CheckScriptFailure(lines, 'E1013: Argument 2: type mismatch, expected number but got bool', 2)
Bram Moolenaare32e5162021-01-21 20:21:29 +0100374enddef
375
Bram Moolenaarefd88552020-06-18 20:50:10 +0200376def Test_missing_return()
Bram Moolenaar62aec932022-01-29 21:45:34 +0000377 v9.CheckDefFailure(['def Missing(): number',
Bram Moolenaarefd88552020-06-18 20:50:10 +0200378 ' if g:cond',
379 ' echo "no return"',
380 ' else',
381 ' return 0',
382 ' endif'
383 'enddef'], 'E1027:')
Bram Moolenaar62aec932022-01-29 21:45:34 +0000384 v9.CheckDefFailure(['def Missing(): number',
Bram Moolenaarefd88552020-06-18 20:50:10 +0200385 ' if g:cond',
386 ' return 1',
387 ' else',
388 ' echo "no return"',
389 ' endif'
390 'enddef'], 'E1027:')
Bram Moolenaar62aec932022-01-29 21:45:34 +0000391 v9.CheckDefFailure(['def Missing(): number',
Bram Moolenaarefd88552020-06-18 20:50:10 +0200392 ' if g:cond',
393 ' return 1',
394 ' else',
395 ' return 2',
396 ' endif'
397 ' return 3'
398 'enddef'], 'E1095:')
399enddef
400
Bram Moolenaar403dc312020-10-17 19:29:51 +0200401def Test_return_bool()
402 var lines =<< trim END
403 vim9script
404 def MenuFilter(id: number, key: string): bool
405 return popup_filter_menu(id, key)
406 enddef
407 def YesnoFilter(id: number, key: string): bool
408 return popup_filter_yesno(id, key)
409 enddef
410 defcompile
411 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000412 v9.CheckScriptSuccess(lines)
Bram Moolenaar403dc312020-10-17 19:29:51 +0200413enddef
414
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200415let s:nothing = 0
416def ReturnNothing()
417 s:nothing = 1
418 if true
419 return
420 endif
421 s:nothing = 2
422enddef
423
424def Test_return_nothing()
Bram Moolenaar62aec932022-01-29 21:45:34 +0000425 g:ReturnNothing()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +0200426 s:nothing->assert_equal(1)
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200427enddef
428
Bram Moolenaar648ea762021-01-15 19:04:32 +0100429def Test_return_invalid()
430 var lines =<< trim END
431 vim9script
432 def Func(): invalid
433 return xxx
434 enddef
435 defcompile
436 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000437 v9.CheckScriptFailure(lines, 'E1010:', 2)
Bram Moolenaar31842cd2021-02-12 22:10:21 +0100438
439 lines =<< trim END
440 vim9script
441 def Test(Fun: func(number): number): list<number>
442 return map([1, 2, 3], (_, i) => Fun(i))
443 enddef
444 defcompile
445 def Inc(nr: number): nr
446 return nr + 2
447 enddef
448 echo Test(Inc)
449 END
450 # doing this twice was leaking memory
Bram Moolenaar62aec932022-01-29 21:45:34 +0000451 v9.CheckScriptFailure(lines, 'E1010:')
452 v9.CheckScriptFailure(lines, 'E1010:')
Bram Moolenaar648ea762021-01-15 19:04:32 +0100453enddef
454
Bram Moolenaarefc084e2021-09-09 22:30:52 +0200455def Test_return_list_any()
Bram Moolenaar114dbda2022-01-03 12:28:03 +0000456 # This used to fail but now the actual list type is checked, and since it has
457 # an item of type string it can be used as list<string>.
Bram Moolenaarefc084e2021-09-09 22:30:52 +0200458 var lines =<< trim END
459 vim9script
460 def Func(): list<string>
461 var l: list<any>
462 l->add('string')
463 return l
464 enddef
465 echo Func()
466 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000467 v9.CheckScriptSuccess(lines)
Bram Moolenaar114dbda2022-01-03 12:28:03 +0000468
Bram Moolenaarefc084e2021-09-09 22:30:52 +0200469 lines =<< trim END
470 vim9script
471 def Func(): list<string>
472 var l: list<any>
473 l += ['string']
474 return l
475 enddef
476 echo Func()
477 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000478 v9.CheckScriptSuccess(lines)
Bram Moolenaarefc084e2021-09-09 22:30:52 +0200479enddef
480
Bram Moolenaar62aec932022-01-29 21:45:34 +0000481func s:Increment()
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200482 let g:counter += 1
483endfunc
484
485def Test_call_ufunc_count()
486 g:counter = 1
487 Increment()
488 Increment()
489 Increment()
Bram Moolenaarf5be8cd2020-07-17 20:36:00 +0200490 # works with and without :call
Bram Moolenaarc0c71e92020-09-11 19:09:48 +0200491 g:counter->assert_equal(4)
492 eval g:counter->assert_equal(4)
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200493 unlet g:counter
494enddef
495
Bram Moolenaar62aec932022-01-29 21:45:34 +0000496def s:MyVarargs(arg: string, ...rest: list<string>): string
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +0200497 var res = arg
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200498 for s in rest
499 res ..= ',' .. s
500 endfor
501 return res
502enddef
503
504def Test_call_varargs()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +0200505 MyVarargs('one')->assert_equal('one')
506 MyVarargs('one', 'two')->assert_equal('one,two')
507 MyVarargs('one', 'two', 'three')->assert_equal('one,two,three')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200508enddef
509
Bram Moolenaar01dd6c32021-09-05 16:36:23 +0200510def Test_call_white_space()
Bram Moolenaar62aec932022-01-29 21:45:34 +0000511 v9.CheckDefAndScriptFailure(["call Test ('text')"], ['E476:', 'E1068:'])
Bram Moolenaar01dd6c32021-09-05 16:36:23 +0200512enddef
513
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200514def MyDefaultArgs(name = 'string'): string
515 return name
516enddef
517
Bram Moolenaar62aec932022-01-29 21:45:34 +0000518def s:MyDefaultSecond(name: string, second: bool = true): string
Bram Moolenaare30f64b2020-07-15 19:48:20 +0200519 return second ? name : 'none'
520enddef
521
Bram Moolenaar38a3bfa2021-03-29 22:14:55 +0200522
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200523def Test_call_default_args()
Bram Moolenaar62aec932022-01-29 21:45:34 +0000524 g:MyDefaultArgs()->assert_equal('string')
525 g:MyDefaultArgs(v:none)->assert_equal('string')
526 g:MyDefaultArgs('one')->assert_equal('one')
527 assert_fails('g:MyDefaultArgs("one", "two")', 'E118:', '', 4, 'Test_call_default_args')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200528
Bram Moolenaarc0c71e92020-09-11 19:09:48 +0200529 MyDefaultSecond('test')->assert_equal('test')
530 MyDefaultSecond('test', true)->assert_equal('test')
531 MyDefaultSecond('test', false)->assert_equal('none')
Bram Moolenaare30f64b2020-07-15 19:48:20 +0200532
Bram Moolenaar38a3bfa2021-03-29 22:14:55 +0200533 var lines =<< trim END
534 def MyDefaultThird(name: string, aa = 'aa', bb = 'bb'): string
535 return name .. aa .. bb
536 enddef
537
538 MyDefaultThird('->')->assert_equal('->aabb')
539 MyDefaultThird('->', v:none)->assert_equal('->aabb')
540 MyDefaultThird('->', 'xx')->assert_equal('->xxbb')
541 MyDefaultThird('->', v:none, v:none)->assert_equal('->aabb')
542 MyDefaultThird('->', 'xx', v:none)->assert_equal('->xxbb')
543 MyDefaultThird('->', v:none, 'yy')->assert_equal('->aayy')
544 MyDefaultThird('->', 'xx', 'yy')->assert_equal('->xxyy')
Bram Moolenaare28d9b32021-07-03 18:56:53 +0200545
546 def DefArg(mandatory: any, optional = mandatory): string
547 return mandatory .. optional
548 enddef
549 DefArg(1234)->assert_equal('12341234')
550 DefArg("ok")->assert_equal('okok')
Bram Moolenaar38a3bfa2021-03-29 22:14:55 +0200551 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000552 v9.CheckDefAndScriptSuccess(lines)
Bram Moolenaar38a3bfa2021-03-29 22:14:55 +0200553
Bram Moolenaar62aec932022-01-29 21:45:34 +0000554 v9.CheckScriptFailure(['def Func(arg: number = asdf)', 'enddef', 'defcompile'], 'E1001:')
Bram Moolenaar2d870f82020-12-05 13:41:01 +0100555 delfunc g:Func
Bram Moolenaar62aec932022-01-29 21:45:34 +0000556 v9.CheckScriptFailure(['def Func(arg: number = "text")', 'enddef', 'defcompile'], 'E1013: Argument 1: type mismatch, expected number but got string')
Bram Moolenaar2d870f82020-12-05 13:41:01 +0100557 delfunc g:Func
Bram Moolenaar62aec932022-01-29 21:45:34 +0000558 v9.CheckDefFailure(['def Func(x: number = )', 'enddef'], 'E15:')
Bram Moolenaar12bce952021-03-11 20:04:04 +0100559
Bram Moolenaar38a3bfa2021-03-29 22:14:55 +0200560 lines =<< trim END
Bram Moolenaar12bce952021-03-11 20:04:04 +0100561 vim9script
562 def Func(a = b == 0 ? 1 : 2, b = 0)
563 enddef
564 defcompile
565 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000566 v9.CheckScriptFailure(lines, 'E1001: Variable not found: b')
Bram Moolenaar59618fe2021-12-21 12:32:17 +0000567
Bram Moolenaarfa46ead2021-12-22 13:18:39 +0000568 # using script variable requires matching type or type cast when executed
Bram Moolenaar59618fe2021-12-21 12:32:17 +0000569 lines =<< trim END
570 vim9script
571 var a: any
572 def Func(arg: string = a)
573 echo arg
574 enddef
575 defcompile
576 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000577 v9.CheckScriptSuccess(lines + ['a = "text"', 'Func()'])
578 v9.CheckScriptFailure(lines + ['a = 123', 'Func()'], 'E1013: Argument 1: type mismatch, expected string but got number')
Bram Moolenaar59618fe2021-12-21 12:32:17 +0000579
580 # using global variable does not require type cast
581 lines =<< trim END
582 vim9script
583 def Func(arg: string = g:str)
584 echo arg
585 enddef
586 g:str = 'works'
587 Func()
588 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000589 v9.CheckScriptSuccess(lines)
Bram Moolenaar04b12692020-05-04 23:24:44 +0200590enddef
591
Bram Moolenaar62aec932022-01-29 21:45:34 +0000592def s:FuncWithComment( # comment
Bram Moolenaarcef12702021-01-04 14:09:43 +0100593 a: number, #comment
594 b: bool, # comment
595 c: string) #comment
596 assert_equal(4, a)
597 assert_equal(true, b)
598 assert_equal('yes', c)
599enddef
600
601def Test_func_with_comments()
602 FuncWithComment(4, true, 'yes')
603
604 var lines =<< trim END
605 def Func(# comment
606 arg: string)
607 enddef
608 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000609 v9.CheckScriptFailure(lines, 'E125:', 1)
Bram Moolenaarcef12702021-01-04 14:09:43 +0100610
611 lines =<< trim END
612 def Func(
613 arg: string# comment
614 )
615 enddef
616 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000617 v9.CheckScriptFailure(lines, 'E475:', 2)
Bram Moolenaarcef12702021-01-04 14:09:43 +0100618
619 lines =<< trim END
620 def Func(
621 arg: string
622 )# comment
623 enddef
624 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000625 v9.CheckScriptFailure(lines, 'E488:', 3)
Bram Moolenaarcef12702021-01-04 14:09:43 +0100626enddef
627
Bram Moolenaar04b12692020-05-04 23:24:44 +0200628def Test_nested_function()
Bram Moolenaar38453522021-11-28 22:00:12 +0000629 def NestedDef(arg: string): string
Bram Moolenaar04b12692020-05-04 23:24:44 +0200630 return 'nested ' .. arg
631 enddef
Bram Moolenaar38453522021-11-28 22:00:12 +0000632 NestedDef(':def')->assert_equal('nested :def')
633
634 func NestedFunc(arg)
635 return 'nested ' .. a:arg
636 endfunc
637 NestedFunc(':func')->assert_equal('nested :func')
Bram Moolenaar04b12692020-05-04 23:24:44 +0200638
Bram Moolenaar62aec932022-01-29 21:45:34 +0000639 v9.CheckDefFailure(['def Nested()', 'enddef', 'Nested(66)'], 'E118:')
640 v9.CheckDefFailure(['def Nested(arg: string)', 'enddef', 'Nested()'], 'E119:')
Bram Moolenaar0e65d3d2020-05-05 17:53:16 +0200641
Bram Moolenaar62aec932022-01-29 21:45:34 +0000642 v9.CheckDefFailure(['def s:Nested()', 'enddef'], 'E1075:')
643 v9.CheckDefFailure(['def b:Nested()', 'enddef'], 'E1075:')
Bram Moolenaar8b848ca2020-09-10 22:28:01 +0200644
Bram Moolenaar54021752020-12-06 18:50:36 +0100645 var lines =<< trim END
646 def Outer()
647 def Inner()
648 # comment
649 enddef
650 def Inner()
651 enddef
652 enddef
653 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000654 v9.CheckDefFailure(lines, 'E1073:')
Bram Moolenaar54021752020-12-06 18:50:36 +0100655
656 lines =<< trim END
657 def Outer()
658 def Inner()
659 # comment
660 enddef
661 def! Inner()
662 enddef
663 enddef
664 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000665 v9.CheckDefFailure(lines, 'E1117:')
Bram Moolenaar54021752020-12-06 18:50:36 +0100666
Bram Moolenaardb8e5c22021-12-25 19:58:22 +0000667 lines =<< trim END
668 vim9script
669 def Outer()
670 def Inner()
671 g:result = 'ok'
672 enddef
673 Inner()
674 enddef
675 Outer()
676 Inner()
677 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000678 v9.CheckScriptFailure(lines, 'E117: Unknown function: Inner')
Bram Moolenaardb8e5c22021-12-25 19:58:22 +0000679 assert_equal('ok', g:result)
680 unlet g:result
681
Bram Moolenaarf681cfb2022-02-07 20:30:57 +0000682 lines =<< trim END
683 vim9script
684 def Outer()
685 def _Inner()
686 echo 'bad'
687 enddef
Bram Moolenaar3787f262022-02-07 21:54:01 +0000688 _Inner()
Bram Moolenaarf681cfb2022-02-07 20:30:57 +0000689 enddef
690 defcompile
691 END
Bram Moolenaar3787f262022-02-07 21:54:01 +0000692 v9.CheckScriptFailure(lines, 'E1267:')
Bram Moolenaarf681cfb2022-02-07 20:30:57 +0000693
694 lines =<< trim END
695 vim9script
696 def Outer()
697 def g:inner()
698 echo 'bad'
699 enddef
Bram Moolenaar3787f262022-02-07 21:54:01 +0000700 g:inner()
Bram Moolenaarf681cfb2022-02-07 20:30:57 +0000701 enddef
702 defcompile
703 END
Bram Moolenaar3787f262022-02-07 21:54:01 +0000704 v9.CheckScriptFailure(lines, 'E1267:')
705
706 lines =<< trim END
707 vim9script
708 def g:_Func()
709 echo 'bad'
710 enddef
711 END
712 v9.CheckScriptFailure(lines, 'E1267:')
713
714 lines =<< trim END
715 vim9script
716 def s:_Func()
717 echo 'bad'
718 enddef
719 END
720 v9.CheckScriptFailure(lines, 'E1267:')
Bram Moolenaarf681cfb2022-02-07 20:30:57 +0000721
Bram Moolenaar54021752020-12-06 18:50:36 +0100722 # nested function inside conditional
Bram Moolenaar54021752020-12-06 18:50:36 +0100723 lines =<< trim END
724 vim9script
725 var thecount = 0
726 if true
727 def Test(): number
728 def TheFunc(): number
729 thecount += 1
730 return thecount
731 enddef
732 return TheFunc()
733 enddef
734 endif
735 defcompile
736 assert_equal(1, Test())
737 assert_equal(2, Test())
738 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000739 v9.CheckScriptSuccess(lines)
Bram Moolenaar8863bda2021-03-17 18:42:08 +0100740
741 # also works when "thecount" is inside the "if" block
742 lines =<< trim END
743 vim9script
744 if true
745 var thecount = 0
746 def Test(): number
747 def TheFunc(): number
748 thecount += 1
749 return thecount
750 enddef
751 return TheFunc()
752 enddef
753 endif
754 defcompile
755 assert_equal(1, Test())
756 assert_equal(2, Test())
757 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000758 v9.CheckScriptSuccess(lines)
Bram Moolenaar4bba16d2021-08-15 19:28:05 +0200759
760 lines =<< trim END
761 vim9script
762 def Outer()
763 def Inner()
764 echo 'hello'
765 enddef burp
766 enddef
767 defcompile
768 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000769 v9.CheckScriptFailure(lines, 'E1173: Text found after enddef: burp', 3)
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200770enddef
771
Bram Moolenaaradc8e442020-12-31 18:28:18 +0100772def Test_not_nested_function()
773 echo printf('%d',
774 function('len')('xxx'))
775enddef
776
Bram Moolenaaraf8edbb2020-08-01 00:03:09 +0200777func Test_call_default_args_from_func()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +0200778 call MyDefaultArgs()->assert_equal('string')
779 call MyDefaultArgs('one')->assert_equal('one')
Bram Moolenaar9bd5d872020-09-06 21:47:48 +0200780 call assert_fails('call MyDefaultArgs("one", "two")', 'E118:', '', 3, 'Test_call_default_args_from_func')
Bram Moolenaaraf8edbb2020-08-01 00:03:09 +0200781endfunc
782
Bram Moolenaar38ddf332020-07-31 22:05:04 +0200783def Test_nested_global_function()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +0200784 var lines =<< trim END
Bram Moolenaar38ddf332020-07-31 22:05:04 +0200785 vim9script
786 def Outer()
787 def g:Inner(): string
788 return 'inner'
789 enddef
790 enddef
Bram Moolenaaraf8edbb2020-08-01 00:03:09 +0200791 defcompile
792 Outer()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +0200793 g:Inner()->assert_equal('inner')
Bram Moolenaaraf8edbb2020-08-01 00:03:09 +0200794 delfunc g:Inner
795 Outer()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +0200796 g:Inner()->assert_equal('inner')
Bram Moolenaaraf8edbb2020-08-01 00:03:09 +0200797 delfunc g:Inner
798 Outer()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +0200799 g:Inner()->assert_equal('inner')
Bram Moolenaaraf8edbb2020-08-01 00:03:09 +0200800 delfunc g:Inner
Bram Moolenaar38ddf332020-07-31 22:05:04 +0200801 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000802 v9.CheckScriptSuccess(lines)
Bram Moolenaar2c79e9d2020-08-01 18:57:52 +0200803
804 lines =<< trim END
805 vim9script
806 def Outer()
Bram Moolenaar38453522021-11-28 22:00:12 +0000807 func g:Inner()
808 return 'inner'
809 endfunc
810 enddef
811 defcompile
812 Outer()
813 g:Inner()->assert_equal('inner')
814 delfunc g:Inner
815 Outer()
816 g:Inner()->assert_equal('inner')
817 delfunc g:Inner
818 Outer()
819 g:Inner()->assert_equal('inner')
820 delfunc g:Inner
821 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000822 v9.CheckScriptSuccess(lines)
Bram Moolenaar38453522021-11-28 22:00:12 +0000823
824 lines =<< trim END
825 vim9script
826 def Outer()
Bram Moolenaar2c79e9d2020-08-01 18:57:52 +0200827 def g:Inner(): string
828 return 'inner'
829 enddef
830 enddef
831 defcompile
832 Outer()
833 Outer()
834 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000835 v9.CheckScriptFailure(lines, "E122:")
Bram Moolenaarcd45ed02020-12-22 17:35:54 +0100836 delfunc g:Inner
Bram Moolenaarad486a02020-08-01 23:22:18 +0200837
838 lines =<< trim END
839 vim9script
Bram Moolenaar58a52f22020-12-22 18:56:55 +0100840 def Outer()
841 def g:Inner()
Bram Moolenaar2949cfd2020-12-31 21:28:47 +0100842 echo map([1, 2, 3], (_, v) => v + 1)
Bram Moolenaar58a52f22020-12-22 18:56:55 +0100843 enddef
844 g:Inner()
845 enddef
846 Outer()
847 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000848 v9.CheckScriptSuccess(lines)
Bram Moolenaar58a52f22020-12-22 18:56:55 +0100849 delfunc g:Inner
850
851 lines =<< trim END
852 vim9script
Bram Moolenaarad486a02020-08-01 23:22:18 +0200853 def Func()
854 echo 'script'
855 enddef
856 def Outer()
857 def Func()
858 echo 'inner'
859 enddef
860 enddef
861 defcompile
862 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000863 v9.CheckScriptFailure(lines, "E1073:", 1)
Bram Moolenaard604d782021-11-20 21:46:20 +0000864
865 lines =<< trim END
866 vim9script
867 def Func()
868 echo 'script'
869 enddef
870 def Func()
871 echo 'script'
872 enddef
873 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000874 v9.CheckScriptFailure(lines, "E1073:", 5)
Bram Moolenaar38ddf332020-07-31 22:05:04 +0200875enddef
876
Bram Moolenaar6abdcf82020-11-22 18:15:44 +0100877def DefListAll()
878 def
879enddef
880
881def DefListOne()
882 def DefListOne
883enddef
884
885def DefListMatches()
886 def /DefList
887enddef
888
889def Test_nested_def_list()
890 var funcs = split(execute('call DefListAll()'), "\n")
891 assert_true(len(funcs) > 10)
892 assert_true(funcs->index('def DefListAll()') >= 0)
893
894 funcs = split(execute('call DefListOne()'), "\n")
895 assert_equal([' def DefListOne()', '1 def DefListOne', ' enddef'], funcs)
896
897 funcs = split(execute('call DefListMatches()'), "\n")
898 assert_true(len(funcs) >= 3)
899 assert_true(funcs->index('def DefListAll()') >= 0)
900 assert_true(funcs->index('def DefListOne()') >= 0)
901 assert_true(funcs->index('def DefListMatches()') >= 0)
Bram Moolenaar54021752020-12-06 18:50:36 +0100902
903 var lines =<< trim END
904 vim9script
905 def Func()
906 def +Func+
907 enddef
908 defcompile
909 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000910 v9.CheckScriptFailure(lines, 'E476:', 1)
Bram Moolenaar6abdcf82020-11-22 18:15:44 +0100911enddef
912
Bram Moolenaar333894b2020-08-01 18:53:07 +0200913def Test_global_local_function()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +0200914 var lines =<< trim END
Bram Moolenaar333894b2020-08-01 18:53:07 +0200915 vim9script
916 def g:Func(): string
917 return 'global'
918 enddef
919 def Func(): string
920 return 'local'
921 enddef
Bram Moolenaarc0c71e92020-09-11 19:09:48 +0200922 g:Func()->assert_equal('global')
923 Func()->assert_equal('local')
Bram Moolenaar2d870f82020-12-05 13:41:01 +0100924 delfunc g:Func
Bram Moolenaar333894b2020-08-01 18:53:07 +0200925 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000926 v9.CheckScriptSuccess(lines)
Bram Moolenaar035d6e92020-08-11 22:30:42 +0200927
928 lines =<< trim END
929 vim9script
930 def g:Funcy()
931 echo 'funcy'
932 enddef
933 s:Funcy()
934 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000935 v9.CheckScriptFailure(lines, 'E117:')
Bram Moolenaar333894b2020-08-01 18:53:07 +0200936enddef
937
Bram Moolenaar0f769812020-09-12 18:32:34 +0200938def Test_local_function_shadows_global()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +0200939 var lines =<< trim END
Bram Moolenaar0f769812020-09-12 18:32:34 +0200940 vim9script
941 def g:Gfunc(): string
942 return 'global'
943 enddef
944 def AnotherFunc(): number
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +0200945 var Gfunc = function('len')
Bram Moolenaar0f769812020-09-12 18:32:34 +0200946 return Gfunc('testing')
947 enddef
948 g:Gfunc()->assert_equal('global')
949 AnotherFunc()->assert_equal(7)
950 delfunc g:Gfunc
951 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000952 v9.CheckScriptSuccess(lines)
Bram Moolenaar0f769812020-09-12 18:32:34 +0200953
954 lines =<< trim END
955 vim9script
956 def g:Func(): string
957 return 'global'
958 enddef
959 def AnotherFunc()
960 g:Func = function('len')
961 enddef
962 AnotherFunc()
963 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000964 v9.CheckScriptFailure(lines, 'E705:')
Bram Moolenaar0f769812020-09-12 18:32:34 +0200965 delfunc g:Func
Bram Moolenaar0865b152021-04-05 15:38:51 +0200966
Bram Moolenaar62aec932022-01-29 21:45:34 +0000967 # global function is not found with g: prefix
Bram Moolenaar0865b152021-04-05 15:38:51 +0200968 lines =<< trim END
969 vim9script
970 def g:Func(): string
971 return 'global'
972 enddef
973 def AnotherFunc(): string
974 return Func()
975 enddef
976 assert_equal('global', AnotherFunc())
Bram Moolenaar0865b152021-04-05 15:38:51 +0200977 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000978 v9.CheckScriptFailure(lines, 'E117:')
979 delfunc g:Func
Bram Moolenaar0865b152021-04-05 15:38:51 +0200980
981 lines =<< trim END
982 vim9script
983 def g:Func(): string
984 return 'global'
985 enddef
Bram Moolenaar848fadd2022-01-30 15:28:30 +0000986 assert_equal('global', g:Func())
Bram Moolenaar0865b152021-04-05 15:38:51 +0200987 delfunc g:Func
988 END
Bram Moolenaar62aec932022-01-29 21:45:34 +0000989 v9.CheckScriptSuccess(lines)
Bram Moolenaar58493cf2022-01-06 12:23:30 +0000990
991 # This does not shadow "i" which is visible only inside the for loop
992 lines =<< trim END
993 vim9script
994
995 def Foo(i: number)
996 echo i
997 enddef
998
999 for i in range(3)
1000 # Foo() is compiled here
1001 Foo(i)
1002 endfor
1003 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001004 v9.CheckScriptSuccess(lines)
Bram Moolenaar0f769812020-09-12 18:32:34 +02001005enddef
1006
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001007func TakesOneArg(arg)
1008 echo a:arg
1009endfunc
1010
1011def Test_call_wrong_args()
Bram Moolenaar62aec932022-01-29 21:45:34 +00001012 v9.CheckDefFailure(['g:TakesOneArg()'], 'E119:')
1013 v9.CheckDefFailure(['g:TakesOneArg(11, 22)'], 'E118:')
1014 v9.CheckDefFailure(['bufnr(xxx)'], 'E1001:')
1015 v9.CheckScriptFailure(['def Func(Ref: func(s: string))'], 'E475:')
Bram Moolenaaree8580e2020-08-28 17:19:07 +02001016
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001017 var lines =<< trim END
Bram Moolenaaree8580e2020-08-28 17:19:07 +02001018 vim9script
1019 def Func(s: string)
1020 echo s
1021 enddef
1022 Func([])
1023 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001024 v9.CheckScriptFailure(lines, 'E1013: Argument 1: type mismatch, expected string but got list<unknown>', 5)
Bram Moolenaarb185a402020-09-18 22:42:00 +02001025
Bram Moolenaar9a015112021-12-31 14:06:45 +00001026 # argument name declared earlier is found when declaring a function
Bram Moolenaarb185a402020-09-18 22:42:00 +02001027 lines =<< trim END
1028 vim9script
Bram Moolenaarb4893b82021-02-21 22:20:24 +01001029 var name = 'piet'
1030 def FuncOne(name: string)
Bram Moolenaar3a5988c2022-02-08 19:23:35 +00001031 echo name
Bram Moolenaarb4893b82021-02-21 22:20:24 +01001032 enddef
1033 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001034 v9.CheckScriptFailure(lines, 'E1168:')
Bram Moolenaarb4893b82021-02-21 22:20:24 +01001035
Bram Moolenaar3a5988c2022-02-08 19:23:35 +00001036 # same, inside the same block
1037 lines =<< trim END
1038 vim9script
1039 if true
1040 var name = 'piet'
1041 def FuncOne(name: string)
1042 echo name
1043 enddef
1044 endif
1045 END
1046 v9.CheckScriptFailure(lines, 'E1168:')
1047
1048 # variable in other block is OK
1049 lines =<< trim END
1050 vim9script
1051 if true
1052 var name = 'piet'
1053 endif
1054 def FuncOne(name: string)
1055 echo name
1056 enddef
1057 END
1058 v9.CheckScriptSuccess(lines)
1059
Bram Moolenaar9a015112021-12-31 14:06:45 +00001060 # argument name declared later is only found when compiling
1061 lines =<< trim END
1062 vim9script
1063 def FuncOne(name: string)
1064 echo nr
1065 enddef
1066 var name = 'piet'
1067 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001068 v9.CheckScriptSuccess(lines)
1069 v9.CheckScriptFailure(lines + ['defcompile'], 'E1168:')
Bram Moolenaar9a015112021-12-31 14:06:45 +00001070
Bram Moolenaarb4893b82021-02-21 22:20:24 +01001071 lines =<< trim END
1072 vim9script
Bram Moolenaarb185a402020-09-18 22:42:00 +02001073 def FuncOne(nr: number)
1074 echo nr
1075 enddef
1076 def FuncTwo()
1077 FuncOne()
1078 enddef
1079 defcompile
1080 END
1081 writefile(lines, 'Xscript')
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001082 var didCatch = false
Bram Moolenaarb185a402020-09-18 22:42:00 +02001083 try
1084 source Xscript
1085 catch
1086 assert_match('E119: Not enough arguments for function: <SNR>\d\+_FuncOne', v:exception)
1087 assert_match('Xscript\[8\]..function <SNR>\d\+_FuncTwo, line 1', v:throwpoint)
1088 didCatch = true
1089 endtry
1090 assert_true(didCatch)
1091
1092 lines =<< trim END
1093 vim9script
1094 def FuncOne(nr: number)
1095 echo nr
1096 enddef
1097 def FuncTwo()
1098 FuncOne(1, 2)
1099 enddef
1100 defcompile
1101 END
1102 writefile(lines, 'Xscript')
1103 didCatch = false
1104 try
1105 source Xscript
1106 catch
1107 assert_match('E118: Too many arguments for function: <SNR>\d\+_FuncOne', v:exception)
1108 assert_match('Xscript\[8\]..function <SNR>\d\+_FuncTwo, line 1', v:throwpoint)
1109 didCatch = true
1110 endtry
1111 assert_true(didCatch)
1112
1113 delete('Xscript')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001114enddef
1115
Bram Moolenaar50824712020-12-20 21:10:17 +01001116def Test_call_funcref_wrong_args()
1117 var head =<< trim END
1118 vim9script
1119 def Func3(a1: string, a2: number, a3: list<number>)
1120 echo a1 .. a2 .. a3[0]
1121 enddef
1122 def Testme()
1123 var funcMap: dict<func> = {func: Func3}
1124 END
1125 var tail =<< trim END
1126 enddef
1127 Testme()
1128 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001129 v9.CheckScriptSuccess(head + ["funcMap['func']('str', 123, [1, 2, 3])"] + tail)
Bram Moolenaar50824712020-12-20 21:10:17 +01001130
Bram Moolenaar62aec932022-01-29 21:45:34 +00001131 v9.CheckScriptFailure(head + ["funcMap['func']('str', 123)"] + tail, 'E119:')
1132 v9.CheckScriptFailure(head + ["funcMap['func']('str', 123, [1], 4)"] + tail, 'E118:')
Bram Moolenaar32b3f822021-01-06 21:59:39 +01001133
1134 var lines =<< trim END
1135 vim9script
1136 var Ref: func(number): any
1137 Ref = (j) => !j
1138 echo Ref(false)
1139 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001140 v9.CheckScriptFailure(lines, 'E1013: Argument 1: type mismatch, expected number but got bool', 4)
Bram Moolenaar32b3f822021-01-06 21:59:39 +01001141
1142 lines =<< trim END
1143 vim9script
1144 var Ref: func(number): any
1145 Ref = (j) => !j
1146 call Ref(false)
1147 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001148 v9.CheckScriptFailure(lines, 'E1013: Argument 1: type mismatch, expected number but got bool', 4)
Bram Moolenaar50824712020-12-20 21:10:17 +01001149enddef
1150
Bram Moolenaarb4d16cb2020-11-05 18:45:46 +01001151def Test_call_lambda_args()
Bram Moolenaar2a389082021-04-09 20:24:31 +02001152 var lines =<< trim END
1153 var Callback = (..._) => 'anything'
1154 assert_equal('anything', Callback())
1155 assert_equal('anything', Callback(1))
1156 assert_equal('anything', Callback('a', 2))
Bram Moolenaar1088b692021-04-09 22:12:44 +02001157
1158 assert_equal('xyz', ((a: string): string => a)('xyz'))
Bram Moolenaar2a389082021-04-09 20:24:31 +02001159 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001160 v9.CheckDefAndScriptSuccess(lines)
Bram Moolenaar2a389082021-04-09 20:24:31 +02001161
Bram Moolenaar62aec932022-01-29 21:45:34 +00001162 v9.CheckDefFailure(['echo ((i) => 0)()'],
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01001163 'E119: Not enough arguments for function: ((i) => 0)()')
Bram Moolenaarb4d16cb2020-11-05 18:45:46 +01001164
Bram Moolenaar2a389082021-04-09 20:24:31 +02001165 lines =<< trim END
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01001166 var Ref = (x: number, y: number) => x + y
Bram Moolenaarb4d16cb2020-11-05 18:45:46 +01001167 echo Ref(1, 'x')
1168 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001169 v9.CheckDefFailure(lines, 'E1013: Argument 2: type mismatch, expected number but got string')
Bram Moolenaare68b02a2021-01-03 13:09:51 +01001170
1171 lines =<< trim END
1172 var Ref: func(job, string, number)
1173 Ref = (x, y) => 0
1174 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001175 v9.CheckDefAndScriptFailure(lines, 'E1012:')
Bram Moolenaare68b02a2021-01-03 13:09:51 +01001176
1177 lines =<< trim END
1178 var Ref: func(job, string)
1179 Ref = (x, y, z) => 0
1180 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001181 v9.CheckDefAndScriptFailure(lines, 'E1012:')
Bram Moolenaar057e84a2021-02-28 16:55:11 +01001182
1183 lines =<< trim END
1184 var one = 1
1185 var l = [1, 2, 3]
1186 echo map(l, (one) => one)
1187 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001188 v9.CheckDefFailure(lines, 'E1167:')
1189 v9.CheckScriptFailure(['vim9script'] + lines, 'E1168:')
Bram Moolenaar057e84a2021-02-28 16:55:11 +01001190
1191 lines =<< trim END
Bram Moolenaar14ded112021-06-26 19:25:49 +02001192 var Ref: func(any, ?any): bool
1193 Ref = (_, y = 1) => false
1194 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001195 v9.CheckDefAndScriptFailure(lines, 'E1172:')
Bram Moolenaar14ded112021-06-26 19:25:49 +02001196
1197 lines =<< trim END
Bram Moolenaar015cf102021-06-26 21:52:02 +02001198 var a = 0
1199 var b = (a == 0 ? 1 : 2)
1200 assert_equal(1, b)
Bram Moolenaar98f9a5f2021-06-26 22:22:38 +02001201 var txt = 'a'
1202 b = (txt =~ 'x' ? 1 : 2)
1203 assert_equal(2, b)
Bram Moolenaar015cf102021-06-26 21:52:02 +02001204 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001205 v9.CheckDefAndScriptSuccess(lines)
Bram Moolenaar015cf102021-06-26 21:52:02 +02001206
1207 lines =<< trim END
Bram Moolenaar057e84a2021-02-28 16:55:11 +01001208 def ShadowLocal()
1209 var one = 1
1210 var l = [1, 2, 3]
1211 echo map(l, (one) => one)
1212 enddef
1213 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001214 v9.CheckDefFailure(lines, 'E1167:')
Bram Moolenaar057e84a2021-02-28 16:55:11 +01001215
1216 lines =<< trim END
1217 def Shadowarg(one: number)
1218 var l = [1, 2, 3]
1219 echo map(l, (one) => one)
1220 enddef
1221 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001222 v9.CheckDefFailure(lines, 'E1167:')
Bram Moolenaar767034c2021-04-09 17:24:52 +02001223
1224 lines =<< trim END
1225 echo ((a) => a)('aa', 'bb')
1226 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001227 v9.CheckDefAndScriptFailure(lines, 'E118:', 1)
Bram Moolenaarc4c56422021-07-21 20:38:46 +02001228
1229 lines =<< trim END
1230 echo 'aa'->((a) => a)('bb')
1231 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001232 v9.CheckDefFailure(lines, 'E118: Too many arguments for function: ->((a) => a)(''bb'')', 1)
1233 v9.CheckScriptFailure(['vim9script'] + lines, 'E118: Too many arguments for function: <lambda>', 2)
Bram Moolenaarb4d16cb2020-11-05 18:45:46 +01001234enddef
1235
Bram Moolenaara755fdb2021-11-20 21:35:41 +00001236def Test_lambda_line_nr()
1237 var lines =<< trim END
1238 vim9script
1239 # comment
1240 # comment
1241 var id = timer_start(1'000, (_) => 0)
1242 var out = execute('verbose ' .. timer_info(id)[0].callback
1243 ->string()
1244 ->substitute("('\\|')", ' ', 'g'))
1245 assert_match('Last set from .* line 4', out)
1246 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001247 v9.CheckScriptSuccess(lines)
Bram Moolenaara755fdb2021-11-20 21:35:41 +00001248enddef
1249
Bram Moolenaar5f91e742021-03-17 21:29:29 +01001250def FilterWithCond(x: string, Cond: func(string): bool): bool
1251 return Cond(x)
1252enddef
1253
Bram Moolenaar0346b792021-01-31 22:18:29 +01001254def Test_lambda_return_type()
1255 var lines =<< trim END
1256 var Ref = (): => 123
1257 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001258 v9.CheckDefAndScriptFailure(lines, 'E1157:', 1)
Bram Moolenaar5f91e742021-03-17 21:29:29 +01001259
Yegappan Lakshmanan611728f2021-05-24 15:15:47 +02001260 # no space before the return type
1261 lines =<< trim END
1262 var Ref = (x):number => x + 1
1263 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001264 v9.CheckDefAndScriptFailure(lines, 'E1069:', 1)
Yegappan Lakshmanan611728f2021-05-24 15:15:47 +02001265
Bram Moolenaar5f91e742021-03-17 21:29:29 +01001266 # this works
1267 for x in ['foo', 'boo']
Bram Moolenaar62aec932022-01-29 21:45:34 +00001268 echo g:FilterWithCond(x, (v) => v =~ '^b')
Bram Moolenaar5f91e742021-03-17 21:29:29 +01001269 endfor
1270
1271 # this fails
1272 lines =<< trim END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001273 echo g:FilterWithCond('foo', (v) => v .. '^b')
Bram Moolenaar5f91e742021-03-17 21:29:29 +01001274 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001275 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 +02001276
1277 lines =<< trim END
1278 var Lambda1 = (x) => {
1279 return x
1280 }
1281 assert_equal('asdf', Lambda1('asdf'))
1282 var Lambda2 = (x): string => {
1283 return x
1284 }
1285 assert_equal('foo', Lambda2('foo'))
1286 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001287 v9.CheckDefAndScriptSuccess(lines)
Bram Moolenaara9931532021-06-12 15:58:16 +02001288
1289 lines =<< trim END
1290 var Lambda = (x): string => {
1291 return x
1292 }
1293 echo Lambda(['foo'])
1294 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001295 v9.CheckDefExecAndScriptFailure(lines, 'E1012:')
Bram Moolenaar0346b792021-01-31 22:18:29 +01001296enddef
1297
Bram Moolenaar709664c2020-12-12 14:33:41 +01001298def Test_lambda_uses_assigned_var()
Bram Moolenaar62aec932022-01-29 21:45:34 +00001299 v9.CheckDefSuccess([
Bram Moolenaar709664c2020-12-12 14:33:41 +01001300 'var x: any = "aaa"'
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01001301 'x = filter(["bbb"], (_, v) => v =~ x)'])
Bram Moolenaar709664c2020-12-12 14:33:41 +01001302enddef
1303
Bram Moolenaar18062fc2021-03-05 21:35:47 +01001304def Test_pass_legacy_lambda_to_def_func()
1305 var lines =<< trim END
1306 vim9script
1307 func Foo()
1308 eval s:Bar({x -> 0})
1309 endfunc
1310 def Bar(y: any)
1311 enddef
1312 Foo()
1313 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001314 v9.CheckScriptSuccess(lines)
Bram Moolenaar831bdf82021-06-22 19:32:17 +02001315
1316 lines =<< trim END
1317 vim9script
Bram Moolenaar7a40ff02021-07-04 15:54:08 +02001318 def g:TestFunc(f: func)
Bram Moolenaar831bdf82021-06-22 19:32:17 +02001319 enddef
1320 legacy call g:TestFunc({-> 0})
1321 delfunc g:TestFunc
1322
1323 def g:TestFunc(f: func(number))
1324 enddef
1325 legacy call g:TestFunc({nr -> 0})
1326 delfunc g:TestFunc
1327 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001328 v9.CheckScriptSuccess(lines)
Bram Moolenaar18062fc2021-03-05 21:35:47 +01001329enddef
1330
Bram Moolenaar844fb642021-10-23 13:32:30 +01001331def Test_lambda_in_reduce_line_break()
1332 # this was using freed memory
1333 var lines =<< trim END
1334 vim9script
1335 const result: dict<number> =
1336 ['Bob', 'Sam', 'Cat', 'Bob', 'Cat', 'Cat']
1337 ->reduce((acc, val) => {
1338 if has_key(acc, val)
1339 acc[val] += 1
1340 return acc
1341 else
1342 acc[val] = 1
1343 return acc
1344 endif
1345 }, {})
1346 assert_equal({Bob: 2, Sam: 1, Cat: 3}, result)
1347 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001348 v9.CheckScriptSuccess(lines)
Bram Moolenaar844fb642021-10-23 13:32:30 +01001349enddef
1350
Bram Moolenaardcb53be2021-12-09 14:23:43 +00001351def Test_set_opfunc_to_lambda()
1352 var lines =<< trim END
1353 vim9script
1354 nnoremap <expr> <F4> <SID>CountSpaces() .. '_'
1355 def CountSpaces(type = ''): string
1356 if type == ''
1357 &operatorfunc = (t) => CountSpaces(t)
1358 return 'g@'
1359 endif
1360 normal! '[V']y
1361 g:result = getreg('"')->count(' ')
1362 return ''
1363 enddef
1364 new
1365 'a b c d e'->setline(1)
1366 feedkeys("\<F4>", 'x')
1367 assert_equal(4, g:result)
1368 bwipe!
1369 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001370 v9.CheckScriptSuccess(lines)
Bram Moolenaardcb53be2021-12-09 14:23:43 +00001371enddef
1372
Bram Moolenaaref082e12021-12-12 21:02:03 +00001373def Test_set_opfunc_to_global_function()
1374 var lines =<< trim END
1375 vim9script
1376 def g:CountSpaces(type = ''): string
1377 normal! '[V']y
1378 g:result = getreg('"')->count(' ')
1379 return ''
1380 enddef
Bram Moolenaarb15cf442021-12-16 15:49:43 +00001381 # global function works at script level
Bram Moolenaaref082e12021-12-12 21:02:03 +00001382 &operatorfunc = g:CountSpaces
1383 new
1384 'a b c d e'->setline(1)
1385 feedkeys("g@_", 'x')
1386 assert_equal(4, g:result)
Bram Moolenaarb15cf442021-12-16 15:49:43 +00001387
1388 &operatorfunc = ''
1389 g:result = 0
1390 # global function works in :def function
1391 def Func()
1392 &operatorfunc = g:CountSpaces
1393 enddef
1394 Func()
1395 feedkeys("g@_", 'x')
1396 assert_equal(4, g:result)
1397
Bram Moolenaaref082e12021-12-12 21:02:03 +00001398 bwipe!
1399 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001400 v9.CheckScriptSuccess(lines)
Bram Moolenaaref082e12021-12-12 21:02:03 +00001401 &operatorfunc = ''
1402enddef
1403
Bram Moolenaar33b968d2021-12-13 11:31:04 +00001404def Test_use_script_func_name_with_prefix()
1405 var lines =<< trim END
1406 vim9script
1407 func s:Getit()
1408 return 'it'
1409 endfunc
1410 var Fn = s:Getit
1411 assert_equal('it', Fn())
1412 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001413 v9.CheckScriptSuccess(lines)
Bram Moolenaar33b968d2021-12-13 11:31:04 +00001414enddef
1415
Bram Moolenaardd297bc2021-12-10 10:37:38 +00001416def Test_lambda_type_allocated()
1417 # Check that unreferencing a partial using a lambda can use the variable type
1418 # after the lambda has been freed and does not leak memory.
1419 var lines =<< trim END
1420 vim9script
1421
1422 func MyomniFunc1(val, findstart, base)
1423 return a:findstart ? 0 : []
1424 endfunc
1425
1426 var Lambda = (a, b) => MyomniFunc1(19, a, b)
1427 &omnifunc = Lambda
1428 Lambda = (a, b) => MyomniFunc1(20, a, b)
1429 &omnifunc = string(Lambda)
1430 Lambda = (a, b) => strlen(a)
1431 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001432 v9.CheckScriptSuccess(lines)
Bram Moolenaardd297bc2021-12-10 10:37:38 +00001433enddef
1434
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001435" Default arg and varargs
1436def MyDefVarargs(one: string, two = 'foo', ...rest: list<string>): string
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001437 var res = one .. ',' .. two
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001438 for s in rest
1439 res ..= ',' .. s
1440 endfor
1441 return res
1442enddef
1443
1444def Test_call_def_varargs()
Bram Moolenaar62aec932022-01-29 21:45:34 +00001445 assert_fails('g:MyDefVarargs()', 'E119:', '', 1, 'Test_call_def_varargs')
1446 g:MyDefVarargs('one')->assert_equal('one,foo')
1447 g:MyDefVarargs('one', 'two')->assert_equal('one,two')
1448 g:MyDefVarargs('one', 'two', 'three')->assert_equal('one,two,three')
1449 v9.CheckDefFailure(['g:MyDefVarargs("one", 22)'],
Bram Moolenaar77072282020-09-16 17:55:40 +02001450 'E1013: Argument 2: type mismatch, expected string but got number')
Bram Moolenaar62aec932022-01-29 21:45:34 +00001451 v9.CheckDefFailure(['g:MyDefVarargs("one", "two", 123)'],
Bram Moolenaar77072282020-09-16 17:55:40 +02001452 'E1013: Argument 3: type mismatch, expected string but got number')
Bram Moolenaar24aa48b2020-07-25 16:33:02 +02001453
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001454 var lines =<< trim END
Bram Moolenaar24aa48b2020-07-25 16:33:02 +02001455 vim9script
1456 def Func(...l: list<string>)
1457 echo l
1458 enddef
1459 Func('a', 'b', 'c')
1460 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001461 v9.CheckScriptSuccess(lines)
Bram Moolenaar24aa48b2020-07-25 16:33:02 +02001462
1463 lines =<< trim END
1464 vim9script
1465 def Func(...l: list<string>)
1466 echo l
1467 enddef
1468 Func()
1469 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001470 v9.CheckScriptSuccess(lines)
Bram Moolenaar24aa48b2020-07-25 16:33:02 +02001471
1472 lines =<< trim END
1473 vim9script
Bram Moolenaar2a389082021-04-09 20:24:31 +02001474 def Func(...l: list<any>)
Bram Moolenaar2f8cbc42020-09-16 17:22:59 +02001475 echo l
1476 enddef
1477 Func(0)
1478 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001479 v9.CheckScriptSuccess(lines)
Bram Moolenaar2f8cbc42020-09-16 17:22:59 +02001480
1481 lines =<< trim END
1482 vim9script
Bram Moolenaar2a389082021-04-09 20:24:31 +02001483 def Func(...l: any)
1484 echo l
1485 enddef
1486 Func(0)
1487 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001488 v9.CheckScriptFailure(lines, 'E1180:', 2)
Bram Moolenaar2a389082021-04-09 20:24:31 +02001489
1490 lines =<< trim END
1491 vim9script
Bram Moolenaar28022722020-09-21 22:02:49 +02001492 def Func(..._l: list<string>)
1493 echo _l
1494 enddef
1495 Func('a', 'b', 'c')
1496 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001497 v9.CheckScriptSuccess(lines)
Bram Moolenaar28022722020-09-21 22:02:49 +02001498
1499 lines =<< trim END
1500 vim9script
Bram Moolenaar24aa48b2020-07-25 16:33:02 +02001501 def Func(...l: list<string>)
1502 echo l
1503 enddef
1504 Func(1, 2, 3)
1505 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001506 v9.CheckScriptFailure(lines, 'E1013: Argument 1: type mismatch')
Bram Moolenaar24aa48b2020-07-25 16:33:02 +02001507
1508 lines =<< trim END
1509 vim9script
1510 def Func(...l: list<string>)
1511 echo l
1512 enddef
1513 Func('a', 9)
1514 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001515 v9.CheckScriptFailure(lines, 'E1013: Argument 2: type mismatch')
Bram Moolenaar24aa48b2020-07-25 16:33:02 +02001516
1517 lines =<< trim END
1518 vim9script
1519 def Func(...l: list<string>)
1520 echo l
1521 enddef
1522 Func(1, 'a')
1523 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001524 v9.CheckScriptFailure(lines, 'E1013: Argument 1: type mismatch')
Bram Moolenaar4f53b792021-02-07 15:59:49 +01001525
1526 lines =<< trim END
1527 vim9script
1528 def Func( # some comment
1529 ...l = []
1530 )
1531 echo l
1532 enddef
1533 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001534 v9.CheckScriptFailure(lines, 'E1160:')
Bram Moolenaar6ce46b92021-08-07 15:35:36 +02001535
1536 lines =<< trim END
1537 vim9script
1538 def DoIt()
1539 g:Later('')
1540 enddef
1541 defcompile
1542 def g:Later(...l: list<number>)
1543 enddef
1544 DoIt()
1545 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001546 v9.CheckScriptFailure(lines, 'E1013: Argument 1: type mismatch, expected number but got string')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001547enddef
1548
Bram Moolenaar1378fbc2020-04-11 20:50:33 +02001549let s:value = ''
1550
1551def FuncOneDefArg(opt = 'text')
1552 s:value = opt
1553enddef
1554
1555def FuncTwoDefArg(nr = 123, opt = 'text'): string
1556 return nr .. opt
1557enddef
1558
1559def FuncVarargs(...arg: list<string>): string
1560 return join(arg, ',')
1561enddef
1562
1563def Test_func_type_varargs()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001564 var RefDefArg: func(?string)
Bram Moolenaar848fadd2022-01-30 15:28:30 +00001565 RefDefArg = g:FuncOneDefArg
Bram Moolenaar1378fbc2020-04-11 20:50:33 +02001566 RefDefArg()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001567 s:value->assert_equal('text')
Bram Moolenaar1378fbc2020-04-11 20:50:33 +02001568 RefDefArg('some')
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001569 s:value->assert_equal('some')
Bram Moolenaar1378fbc2020-04-11 20:50:33 +02001570
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001571 var RefDef2Arg: func(?number, ?string): string
Bram Moolenaar848fadd2022-01-30 15:28:30 +00001572 RefDef2Arg = g:FuncTwoDefArg
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001573 RefDef2Arg()->assert_equal('123text')
1574 RefDef2Arg(99)->assert_equal('99text')
1575 RefDef2Arg(77, 'some')->assert_equal('77some')
Bram Moolenaar1378fbc2020-04-11 20:50:33 +02001576
Bram Moolenaar62aec932022-01-29 21:45:34 +00001577 v9.CheckDefFailure(['var RefWrong: func(string?)'], 'E1010:')
1578 v9.CheckDefFailure(['var RefWrong: func(?string, string)'], 'E1007:')
Bram Moolenaar1378fbc2020-04-11 20:50:33 +02001579
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001580 var RefVarargs: func(...list<string>): string
Bram Moolenaar848fadd2022-01-30 15:28:30 +00001581 RefVarargs = g:FuncVarargs
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001582 RefVarargs()->assert_equal('')
1583 RefVarargs('one')->assert_equal('one')
1584 RefVarargs('one', 'two')->assert_equal('one,two')
Bram Moolenaar1378fbc2020-04-11 20:50:33 +02001585
Bram Moolenaar62aec932022-01-29 21:45:34 +00001586 v9.CheckDefFailure(['var RefWrong: func(...list<string>, string)'], 'E110:')
1587 v9.CheckDefFailure(['var RefWrong: func(...list<string>, ?string)'], 'E110:')
Bram Moolenaar1378fbc2020-04-11 20:50:33 +02001588enddef
1589
Bram Moolenaar0b76b422020-04-07 22:05:08 +02001590" Only varargs
1591def MyVarargsOnly(...args: list<string>): string
1592 return join(args, ',')
1593enddef
1594
1595def Test_call_varargs_only()
Bram Moolenaar62aec932022-01-29 21:45:34 +00001596 g:MyVarargsOnly()->assert_equal('')
1597 g:MyVarargsOnly('one')->assert_equal('one')
1598 g:MyVarargsOnly('one', 'two')->assert_equal('one,two')
1599 v9.CheckDefFailure(['g:MyVarargsOnly(1)'], 'E1013: Argument 1: type mismatch, expected string but got number')
1600 v9.CheckDefFailure(['g:MyVarargsOnly("one", 2)'], 'E1013: Argument 2: type mismatch, expected string but got number')
Bram Moolenaar0b76b422020-04-07 22:05:08 +02001601enddef
1602
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001603def Test_using_var_as_arg()
Bram Moolenaard2939812021-12-30 17:09:05 +00001604 var lines =<< trim END
1605 def Func(x: number)
1606 var x = 234
1607 enddef
1608 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001609 v9.CheckDefFailure(lines, 'E1006:')
Bram Moolenaard2939812021-12-30 17:09:05 +00001610
1611 lines =<< trim END
1612 def Func(Ref: number)
1613 def Ref()
1614 enddef
1615 enddef
1616 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001617 v9.CheckDefFailure(lines, 'E1073:')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001618enddef
1619
Bram Moolenaar62aec932022-01-29 21:45:34 +00001620def s:DictArg(arg: dict<string>)
Bram Moolenaarcb2bdb12020-05-10 22:53:56 +02001621 arg['key'] = 'value'
1622enddef
1623
Bram Moolenaar62aec932022-01-29 21:45:34 +00001624def s:ListArg(arg: list<string>)
Bram Moolenaarcb2bdb12020-05-10 22:53:56 +02001625 arg[0] = 'value'
1626enddef
1627
1628def Test_assign_to_argument()
Bram Moolenaarf5be8cd2020-07-17 20:36:00 +02001629 # works for dict and list
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001630 var d: dict<string> = {}
Bram Moolenaarcb2bdb12020-05-10 22:53:56 +02001631 DictArg(d)
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001632 d['key']->assert_equal('value')
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001633 var l: list<string> = []
Bram Moolenaarcb2bdb12020-05-10 22:53:56 +02001634 ListArg(l)
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001635 l[0]->assert_equal('value')
Bram Moolenaarcb2bdb12020-05-10 22:53:56 +02001636
Bram Moolenaar62aec932022-01-29 21:45:34 +00001637 v9.CheckScriptFailure(['def Func(arg: number)', 'arg = 3', 'enddef', 'defcompile'], 'E1090:')
Bram Moolenaar2d870f82020-12-05 13:41:01 +01001638 delfunc! g:Func
Bram Moolenaarcb2bdb12020-05-10 22:53:56 +02001639enddef
1640
Bram Moolenaarb816dae2020-09-20 22:04:00 +02001641" These argument names are reserved in legacy functions.
Bram Moolenaar62aec932022-01-29 21:45:34 +00001642def s:WithReservedNames(firstline: string, lastline: string): string
Bram Moolenaarb816dae2020-09-20 22:04:00 +02001643 return firstline .. lastline
1644enddef
1645
1646def Test_argument_names()
1647 assert_equal('OK', WithReservedNames('O', 'K'))
1648enddef
1649
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001650def Test_call_func_defined_later()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001651 g:DefinedLater('one')->assert_equal('one')
Bram Moolenaar9bd5d872020-09-06 21:47:48 +02001652 assert_fails('NotDefined("one")', 'E117:', '', 2, 'Test_call_func_defined_later')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001653enddef
1654
Bram Moolenaar1df8b3f2020-04-23 18:13:23 +02001655func DefinedLater(arg)
1656 return a:arg
1657endfunc
1658
1659def Test_call_funcref()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001660 g:SomeFunc('abc')->assert_equal(3)
Bram Moolenaar9bd5d872020-09-06 21:47:48 +02001661 assert_fails('NotAFunc()', 'E117:', '', 2, 'Test_call_funcref') # comment after call
Bram Moolenaar2ef91562021-12-11 16:14:07 +00001662 assert_fails('g:NotAFunc()', 'E1085:', '', 3, 'Test_call_funcref')
Bram Moolenaar2f1980f2020-07-22 19:30:06 +02001663
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001664 var lines =<< trim END
Bram Moolenaar2f1980f2020-07-22 19:30:06 +02001665 vim9script
1666 def RetNumber(): number
1667 return 123
1668 enddef
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001669 var Funcref: func: number = function('RetNumber')
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001670 Funcref()->assert_equal(123)
Bram Moolenaar2f1980f2020-07-22 19:30:06 +02001671 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001672 v9.CheckScriptSuccess(lines)
Bram Moolenaar0f60e802020-07-22 20:16:11 +02001673
1674 lines =<< trim END
1675 vim9script
1676 def RetNumber(): number
1677 return 123
1678 enddef
1679 def Bar(F: func: number): number
1680 return F()
1681 enddef
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001682 var Funcref = function('RetNumber')
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001683 Bar(Funcref)->assert_equal(123)
Bram Moolenaar0f60e802020-07-22 20:16:11 +02001684 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001685 v9.CheckScriptSuccess(lines)
Bram Moolenaarbfba8652020-07-23 20:09:10 +02001686
1687 lines =<< trim END
1688 vim9script
1689 def UseNumber(nr: number)
1690 echo nr
1691 enddef
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001692 var Funcref: func(number) = function('UseNumber')
Bram Moolenaarbfba8652020-07-23 20:09:10 +02001693 Funcref(123)
1694 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001695 v9.CheckScriptSuccess(lines)
Bram Moolenaarb8070e32020-07-23 20:56:04 +02001696
1697 lines =<< trim END
1698 vim9script
1699 def UseNumber(nr: number)
1700 echo nr
1701 enddef
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001702 var Funcref: func(string) = function('UseNumber')
Bram Moolenaarb8070e32020-07-23 20:56:04 +02001703 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001704 v9.CheckScriptFailure(lines, 'E1012: Type mismatch; expected func(string) but got func(number)')
Bram Moolenaar4fc224c2020-07-26 17:56:25 +02001705
1706 lines =<< trim END
1707 vim9script
1708 def EchoNr(nr = 34)
1709 g:echo = nr
1710 enddef
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001711 var Funcref: func(?number) = function('EchoNr')
Bram Moolenaar4fc224c2020-07-26 17:56:25 +02001712 Funcref()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001713 g:echo->assert_equal(34)
Bram Moolenaar4fc224c2020-07-26 17:56:25 +02001714 Funcref(123)
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001715 g:echo->assert_equal(123)
Bram Moolenaar4fc224c2020-07-26 17:56:25 +02001716 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001717 v9.CheckScriptSuccess(lines)
Bram Moolenaarace61322020-07-26 18:16:58 +02001718
1719 lines =<< trim END
1720 vim9script
1721 def EchoList(...l: list<number>)
1722 g:echo = l
1723 enddef
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001724 var Funcref: func(...list<number>) = function('EchoList')
Bram Moolenaarace61322020-07-26 18:16:58 +02001725 Funcref()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001726 g:echo->assert_equal([])
Bram Moolenaarace61322020-07-26 18:16:58 +02001727 Funcref(1, 2, 3)
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001728 g:echo->assert_equal([1, 2, 3])
Bram Moolenaarace61322020-07-26 18:16:58 +02001729 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001730 v9.CheckScriptSuccess(lines)
Bram Moolenaar01865ad2020-07-26 18:33:09 +02001731
1732 lines =<< trim END
1733 vim9script
1734 def OptAndVar(nr: number, opt = 12, ...l: list<number>): number
1735 g:optarg = opt
1736 g:listarg = l
1737 return nr
1738 enddef
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001739 var Funcref: func(number, ?number, ...list<number>): number = function('OptAndVar')
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001740 Funcref(10)->assert_equal(10)
1741 g:optarg->assert_equal(12)
1742 g:listarg->assert_equal([])
Bram Moolenaar01865ad2020-07-26 18:33:09 +02001743
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001744 Funcref(11, 22)->assert_equal(11)
1745 g:optarg->assert_equal(22)
1746 g:listarg->assert_equal([])
Bram Moolenaar01865ad2020-07-26 18:33:09 +02001747
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001748 Funcref(17, 18, 1, 2, 3)->assert_equal(17)
1749 g:optarg->assert_equal(18)
1750 g:listarg->assert_equal([1, 2, 3])
Bram Moolenaar01865ad2020-07-26 18:33:09 +02001751 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001752 v9.CheckScriptSuccess(lines)
Bram Moolenaar1df8b3f2020-04-23 18:13:23 +02001753enddef
1754
1755let SomeFunc = function('len')
1756let NotAFunc = 'text'
1757
Bram Moolenaar99aaf0c2020-04-12 14:39:53 +02001758def CombineFuncrefTypes()
Bram Moolenaarf5be8cd2020-07-17 20:36:00 +02001759 # same arguments, different return type
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001760 var Ref1: func(bool): string
1761 var Ref2: func(bool): number
1762 var Ref3: func(bool): any
Bram Moolenaar99aaf0c2020-04-12 14:39:53 +02001763 Ref3 = g:cond ? Ref1 : Ref2
1764
Bram Moolenaarf5be8cd2020-07-17 20:36:00 +02001765 # different number of arguments
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001766 var Refa1: func(bool): number
1767 var Refa2: func(bool, number): number
1768 var Refa3: func: number
Bram Moolenaar99aaf0c2020-04-12 14:39:53 +02001769 Refa3 = g:cond ? Refa1 : Refa2
1770
Bram Moolenaarf5be8cd2020-07-17 20:36:00 +02001771 # different argument types
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001772 var Refb1: func(bool, string): number
1773 var Refb2: func(string, number): number
1774 var Refb3: func(any, any): number
Bram Moolenaar99aaf0c2020-04-12 14:39:53 +02001775 Refb3 = g:cond ? Refb1 : Refb2
1776enddef
1777
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001778def FuncWithForwardCall()
Bram Moolenaar1df8b3f2020-04-23 18:13:23 +02001779 return g:DefinedEvenLater("yes")
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001780enddef
1781
1782def DefinedEvenLater(arg: string): string
1783 return arg
1784enddef
1785
1786def Test_error_in_nested_function()
Bram Moolenaarf5be8cd2020-07-17 20:36:00 +02001787 # Error in called function requires unwinding the call stack.
Bram Moolenaar848fadd2022-01-30 15:28:30 +00001788 assert_fails('g:FuncWithForwardCall()', 'E1096:', '', 1, 'FuncWithForwardCall')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001789enddef
1790
Bram Moolenaar4bf10062021-12-28 17:23:12 +00001791def Test_nested_function_with_nextcmd()
Bram Moolenaar9c23f9b2021-12-26 14:23:22 +00001792 var lines =<< trim END
1793 vim9script
1794 # Define an outer function
1795 def FirstFunction()
1796 # Define an inner function
1797 def SecondFunction()
1798 # the function has a body, a double free is detected.
1799 AAAAA
1800
1801 # enddef followed by | or } followed by # one or more characters
1802 enddef|BBBB
1803 enddef
1804
1805 # Compile all functions
1806 defcompile
1807 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001808 v9.CheckScriptFailure(lines, 'E1173: Text found after enddef: BBBB')
Bram Moolenaar9c23f9b2021-12-26 14:23:22 +00001809enddef
1810
Bram Moolenaar4bf10062021-12-28 17:23:12 +00001811def Test_nested_function_with_args_split()
1812 var lines =<< trim END
1813 vim9script
1814 def FirstFunction()
1815 def SecondFunction(
1816 )
1817 # had a double free if the right parenthesis of the nested function is
1818 # on the next line
1819
1820 enddef|BBBB
1821 enddef
1822 # Compile all functions
1823 defcompile
1824 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001825 v9.CheckScriptFailure(lines, 'E1173: Text found after enddef: BBBB')
Bram Moolenaar7473a842021-12-28 17:55:26 +00001826
1827 lines =<< trim END
1828 vim9script
1829 def FirstFunction()
1830 func SecondFunction()
1831 endfunc|BBBB
1832 enddef
1833 defcompile
1834 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001835 v9.CheckScriptFailure(lines, 'E1173: Text found after endfunction: BBBB')
Bram Moolenaar4bf10062021-12-28 17:23:12 +00001836enddef
1837
Bram Moolenaar9f1a39a2022-01-08 15:39:39 +00001838def Test_error_in_function_args()
1839 var lines =<< trim END
1840 def FirstFunction()
1841 def SecondFunction(J =
1842 # Nois
1843 # one
1844
1845 enddef|BBBB
1846 enddef
1847 # Compile all functions
1848 defcompile
1849 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001850 v9.CheckScriptFailure(lines, 'E488:')
Bram Moolenaar9f1a39a2022-01-08 15:39:39 +00001851enddef
1852
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001853def Test_return_type_wrong()
Bram Moolenaar62aec932022-01-29 21:45:34 +00001854 v9.CheckScriptFailure([
Bram Moolenaar5a849da2020-08-08 16:47:30 +02001855 'def Func(): number',
1856 'return "a"',
1857 'enddef',
1858 'defcompile'], 'expected number but got string')
Bram Moolenaar2d870f82020-12-05 13:41:01 +01001859 delfunc! g:Func
Bram Moolenaar62aec932022-01-29 21:45:34 +00001860 v9.CheckScriptFailure([
Bram Moolenaar5a849da2020-08-08 16:47:30 +02001861 'def Func(): string',
1862 'return 1',
1863 'enddef',
1864 'defcompile'], 'expected string but got number')
Bram Moolenaar2d870f82020-12-05 13:41:01 +01001865 delfunc! g:Func
Bram Moolenaar62aec932022-01-29 21:45:34 +00001866 v9.CheckScriptFailure([
Bram Moolenaar5a849da2020-08-08 16:47:30 +02001867 'def Func(): void',
1868 'return "a"',
1869 'enddef',
1870 'defcompile'],
1871 'E1096: Returning a value in a function without a return type')
Bram Moolenaar2d870f82020-12-05 13:41:01 +01001872 delfunc! g:Func
Bram Moolenaar62aec932022-01-29 21:45:34 +00001873 v9.CheckScriptFailure([
Bram Moolenaar5a849da2020-08-08 16:47:30 +02001874 'def Func()',
1875 'return "a"',
1876 'enddef',
1877 'defcompile'],
1878 'E1096: Returning a value in a function without a return type')
Bram Moolenaar2d870f82020-12-05 13:41:01 +01001879 delfunc! g:Func
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001880
Bram Moolenaar62aec932022-01-29 21:45:34 +00001881 v9.CheckScriptFailure([
Bram Moolenaar5a849da2020-08-08 16:47:30 +02001882 'def Func(): number',
1883 'return',
1884 'enddef',
1885 'defcompile'], 'E1003:')
Bram Moolenaar2d870f82020-12-05 13:41:01 +01001886 delfunc! g:Func
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001887
Bram Moolenaar62aec932022-01-29 21:45:34 +00001888 v9.CheckScriptFailure([
Bram Moolenaar33ea9fd2021-08-08 19:07:37 +02001889 'def Func():number',
1890 'return 123',
1891 'enddef',
1892 'defcompile'], 'E1069:')
1893 delfunc! g:Func
1894
Bram Moolenaar62aec932022-01-29 21:45:34 +00001895 v9.CheckScriptFailure([
Bram Moolenaar33ea9fd2021-08-08 19:07:37 +02001896 'def Func() :number',
1897 'return 123',
1898 'enddef',
1899 'defcompile'], 'E1059:')
1900 delfunc! g:Func
1901
Bram Moolenaar62aec932022-01-29 21:45:34 +00001902 v9.CheckScriptFailure([
Bram Moolenaar33ea9fd2021-08-08 19:07:37 +02001903 'def Func() : number',
1904 'return 123',
1905 'enddef',
1906 'defcompile'], 'E1059:')
1907 delfunc! g:Func
1908
Bram Moolenaar62aec932022-01-29 21:45:34 +00001909 v9.CheckScriptFailure(['def Func(): list', 'return []', 'enddef'], 'E1008:')
Bram Moolenaar2d870f82020-12-05 13:41:01 +01001910 delfunc! g:Func
Bram Moolenaar62aec932022-01-29 21:45:34 +00001911 v9.CheckScriptFailure(['def Func(): dict', 'return {}', 'enddef'], 'E1008:')
Bram Moolenaar2d870f82020-12-05 13:41:01 +01001912 delfunc! g:Func
Bram Moolenaar62aec932022-01-29 21:45:34 +00001913 v9.CheckScriptFailure(['def Func()', 'return 1'], 'E1057:')
Bram Moolenaar2d870f82020-12-05 13:41:01 +01001914 delfunc! g:Func
Bram Moolenaar5a849da2020-08-08 16:47:30 +02001915
Bram Moolenaar62aec932022-01-29 21:45:34 +00001916 v9.CheckScriptFailure([
Bram Moolenaar5a849da2020-08-08 16:47:30 +02001917 'vim9script',
1918 'def FuncB()',
1919 ' return 123',
1920 'enddef',
1921 'def FuncA()',
1922 ' FuncB()',
1923 'enddef',
1924 'defcompile'], 'E1096:')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001925enddef
1926
1927def Test_arg_type_wrong()
Bram Moolenaar62aec932022-01-29 21:45:34 +00001928 v9.CheckScriptFailure(['def Func3(items: list)', 'echo "a"', 'enddef'], 'E1008: Missing <type>')
1929 v9.CheckScriptFailure(['def Func4(...)', 'echo "a"', 'enddef'], 'E1055: Missing name after ...')
1930 v9.CheckScriptFailure(['def Func5(items:string)', 'echo "a"'], 'E1069:')
1931 v9.CheckScriptFailure(['def Func5(items)', 'echo "a"'], 'E1077:')
1932 v9.CheckScriptFailure(['def Func6(...x:list<number>)', 'echo "a"', 'enddef'], 'E1069:')
1933 v9.CheckScriptFailure(['def Func7(...x: int)', 'echo "a"', 'enddef'], 'E1010:')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001934enddef
1935
Bram Moolenaar86cdb8a2021-04-06 19:01:03 +02001936def Test_white_space_before_comma()
1937 var lines =<< trim END
1938 vim9script
1939 def Func(a: number , b: number)
1940 enddef
1941 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001942 v9.CheckScriptFailure(lines, 'E1068:')
Yegappan Lakshmanan611728f2021-05-24 15:15:47 +02001943 call assert_fails('vim9cmd echo stridx("a" .. "b" , "a")', 'E1068:')
Bram Moolenaar86cdb8a2021-04-06 19:01:03 +02001944enddef
1945
Bram Moolenaar608d78f2021-03-06 22:33:12 +01001946def Test_white_space_after_comma()
1947 var lines =<< trim END
1948 vim9script
1949 def Func(a: number,b: number)
1950 enddef
1951 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001952 v9.CheckScriptFailure(lines, 'E1069:')
Bram Moolenaar608d78f2021-03-06 22:33:12 +01001953
1954 # OK in legacy function
1955 lines =<< trim END
1956 vim9script
1957 func Func(a,b)
1958 endfunc
1959 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00001960 v9.CheckScriptSuccess(lines)
Bram Moolenaar608d78f2021-03-06 22:33:12 +01001961enddef
1962
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001963def Test_vim9script_call()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001964 var lines =<< trim END
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001965 vim9script
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001966 var name = ''
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001967 def MyFunc(arg: string)
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001968 name = arg
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001969 enddef
1970 MyFunc('foobar')
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001971 name->assert_equal('foobar')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001972
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001973 var str = 'barfoo'
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001974 str->MyFunc()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001975 name->assert_equal('barfoo')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001976
Bram Moolenaar67979662020-06-20 22:50:47 +02001977 g:value = 'value'
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001978 g:value->MyFunc()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001979 name->assert_equal('value')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001980
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001981 var listvar = []
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001982 def ListFunc(arg: list<number>)
1983 listvar = arg
1984 enddef
1985 [1, 2, 3]->ListFunc()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001986 listvar->assert_equal([1, 2, 3])
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001987
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001988 var dictvar = {}
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001989 def DictFunc(arg: dict<number>)
1990 dictvar = arg
1991 enddef
Bram Moolenaare0de1712020-12-02 17:36:54 +01001992 {a: 1, b: 2}->DictFunc()
1993 dictvar->assert_equal({a: 1, b: 2})
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001994 def CompiledDict()
Bram Moolenaare0de1712020-12-02 17:36:54 +01001995 {a: 3, b: 4}->DictFunc()
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001996 enddef
1997 CompiledDict()
Bram Moolenaare0de1712020-12-02 17:36:54 +01001998 dictvar->assert_equal({a: 3, b: 4})
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001999
Bram Moolenaare0de1712020-12-02 17:36:54 +01002000 {a: 3, b: 4}->DictFunc()
2001 dictvar->assert_equal({a: 3, b: 4})
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002002
2003 ('text')->MyFunc()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002004 name->assert_equal('text')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002005 ("some")->MyFunc()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002006 name->assert_equal('some')
Bram Moolenaare6b53242020-07-01 17:28:33 +02002007
Bram Moolenaar13e12b82020-07-24 18:47:22 +02002008 # line starting with single quote is not a mark
Bram Moolenaar10409562020-07-29 20:00:38 +02002009 # line starting with double quote can be a method call
Bram Moolenaar3d48e252020-07-15 14:15:52 +02002010 'asdfasdf'->MyFunc()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002011 name->assert_equal('asdfasdf')
Bram Moolenaar10409562020-07-29 20:00:38 +02002012 "xyz"->MyFunc()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002013 name->assert_equal('xyz')
Bram Moolenaar3d48e252020-07-15 14:15:52 +02002014
2015 def UseString()
2016 'xyork'->MyFunc()
2017 enddef
2018 UseString()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002019 name->assert_equal('xyork')
Bram Moolenaar3d48e252020-07-15 14:15:52 +02002020
Bram Moolenaar10409562020-07-29 20:00:38 +02002021 def UseString2()
2022 "knife"->MyFunc()
2023 enddef
2024 UseString2()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002025 name->assert_equal('knife')
Bram Moolenaar10409562020-07-29 20:00:38 +02002026
Bram Moolenaar13e12b82020-07-24 18:47:22 +02002027 # prepending a colon makes it a mark
2028 new
2029 setline(1, ['aaa', 'bbb', 'ccc'])
2030 normal! 3Gmt1G
2031 :'t
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002032 getcurpos()[1]->assert_equal(3)
Bram Moolenaar13e12b82020-07-24 18:47:22 +02002033 bwipe!
2034
Bram Moolenaare6b53242020-07-01 17:28:33 +02002035 MyFunc(
2036 'continued'
2037 )
2038 assert_equal('continued',
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002039 name
Bram Moolenaare6b53242020-07-01 17:28:33 +02002040 )
2041
2042 call MyFunc(
2043 'more'
2044 ..
2045 'lines'
2046 )
2047 assert_equal(
2048 'morelines',
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002049 name)
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002050 END
2051 writefile(lines, 'Xcall.vim')
2052 source Xcall.vim
2053 delete('Xcall.vim')
2054enddef
2055
2056def Test_vim9script_call_fail_decl()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002057 var lines =<< trim END
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002058 vim9script
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002059 var name = ''
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002060 def MyFunc(arg: string)
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002061 var name = 123
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002062 enddef
Bram Moolenaar822ba242020-05-24 23:00:18 +02002063 defcompile
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002064 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002065 v9.CheckScriptFailure(lines, 'E1054:')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002066enddef
2067
Bram Moolenaar65b95452020-07-19 14:03:09 +02002068def Test_vim9script_call_fail_type()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002069 var lines =<< trim END
Bram Moolenaar65b95452020-07-19 14:03:09 +02002070 vim9script
2071 def MyFunc(arg: string)
2072 echo arg
2073 enddef
2074 MyFunc(1234)
2075 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002076 v9.CheckScriptFailure(lines, 'E1013: Argument 1: type mismatch, expected string but got number')
Bram Moolenaar65b95452020-07-19 14:03:09 +02002077enddef
2078
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002079def Test_vim9script_call_fail_const()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002080 var lines =<< trim END
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002081 vim9script
2082 const var = ''
2083 def MyFunc(arg: string)
2084 var = 'asdf'
2085 enddef
Bram Moolenaar822ba242020-05-24 23:00:18 +02002086 defcompile
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002087 END
2088 writefile(lines, 'Xcall_const.vim')
Bram Moolenaar9bd5d872020-09-06 21:47:48 +02002089 assert_fails('source Xcall_const.vim', 'E46:', '', 1, 'MyFunc')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002090 delete('Xcall_const.vim')
Bram Moolenaar3bdc90b2020-12-22 20:35:40 +01002091
2092 lines =<< trim END
2093 const g:Aconst = 77
2094 def Change()
2095 # comment
2096 g:Aconst = 99
2097 enddef
2098 call Change()
2099 unlet g:Aconst
2100 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002101 v9.CheckScriptFailure(lines, 'E741: Value is locked: Aconst', 2)
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002102enddef
2103
2104" Test that inside :function a Python function can be defined, :def is not
2105" recognized.
2106func Test_function_python()
2107 CheckFeature python3
Bram Moolenaar727345e2020-09-27 23:33:59 +02002108 let py = 'python3'
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002109 execute py "<< EOF"
2110def do_something():
2111 return 1
2112EOF
2113endfunc
2114
2115def Test_delfunc()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002116 var lines =<< trim END
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002117 vim9script
Bram Moolenaar4c17ad92020-04-27 22:47:51 +02002118 def g:GoneSoon()
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002119 echo 'hello'
2120 enddef
2121
2122 def CallGoneSoon()
Bram Moolenaar62aec932022-01-29 21:45:34 +00002123 g:GoneSoon()
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002124 enddef
Bram Moolenaar822ba242020-05-24 23:00:18 +02002125 defcompile
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002126
Bram Moolenaar4c17ad92020-04-27 22:47:51 +02002127 delfunc g:GoneSoon
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002128 CallGoneSoon()
2129 END
2130 writefile(lines, 'XToDelFunc')
Bram Moolenaar9bd5d872020-09-06 21:47:48 +02002131 assert_fails('so XToDelFunc', 'E933:', '', 1, 'CallGoneSoon')
2132 assert_fails('so XToDelFunc', 'E933:', '', 1, 'CallGoneSoon')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002133
2134 delete('XToDelFunc')
2135enddef
2136
Bram Moolenaar7509ad82021-12-14 18:14:37 +00002137func Test_free_dict_while_in_funcstack()
2138 " relies on the sleep command
2139 CheckUnix
2140 call Run_Test_free_dict_while_in_funcstack()
2141endfunc
2142
2143def Run_Test_free_dict_while_in_funcstack()
Bram Moolenaar7509ad82021-12-14 18:14:37 +00002144 # this was freeing the TermRun() default argument dictionary while it was
2145 # still referenced in a funcstack_T
2146 var lines =<< trim END
2147 vim9script
2148
2149 &updatetime = 400
2150 def TermRun(_ = {})
2151 def Post()
2152 enddef
2153 def Exec()
2154 term_start('sleep 1', {
2155 term_finish: 'close',
2156 exit_cb: (_, _) => Post(),
2157 })
2158 enddef
2159 Exec()
2160 enddef
2161 nnoremap <F4> <Cmd>call <SID>TermRun()<CR>
2162 timer_start(100, (_) => feedkeys("\<F4>"))
2163 timer_start(1000, (_) => feedkeys("\<F4>"))
2164 sleep 1500m
2165 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002166 v9.CheckScriptSuccess(lines)
Bram Moolenaar7509ad82021-12-14 18:14:37 +00002167 nunmap <F4>
2168 set updatetime&
2169enddef
2170
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002171def Test_redef_failure()
Bram Moolenaard2c61702020-09-06 15:58:36 +02002172 writefile(['def Func0(): string', 'return "Func0"', 'enddef'], 'Xdef')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002173 so Xdef
Bram Moolenaard2c61702020-09-06 15:58:36 +02002174 writefile(['def Func1(): string', 'return "Func1"', 'enddef'], 'Xdef')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002175 so Xdef
Bram Moolenaard2c61702020-09-06 15:58:36 +02002176 writefile(['def! Func0(): string', 'enddef', 'defcompile'], 'Xdef')
Bram Moolenaar9bd5d872020-09-06 21:47:48 +02002177 assert_fails('so Xdef', 'E1027:', '', 1, 'Func0')
Bram Moolenaard2c61702020-09-06 15:58:36 +02002178 writefile(['def Func2(): string', 'return "Func2"', 'enddef'], 'Xdef')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002179 so Xdef
Bram Moolenaard2c61702020-09-06 15:58:36 +02002180 delete('Xdef')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002181
Bram Moolenaar701cc6c2021-04-10 13:33:48 +02002182 assert_fails('g:Func0()', 'E1091:')
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002183 g:Func1()->assert_equal('Func1')
2184 g:Func2()->assert_equal('Func2')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002185
2186 delfunc! Func0
2187 delfunc! Func1
2188 delfunc! Func2
2189enddef
2190
Bram Moolenaarf93c7fe2020-04-23 22:16:53 +02002191def Test_vim9script_func()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002192 var lines =<< trim END
Bram Moolenaarf93c7fe2020-04-23 22:16:53 +02002193 vim9script
2194 func Func(arg)
2195 echo a:arg
2196 endfunc
2197 Func('text')
2198 END
2199 writefile(lines, 'XVim9Func')
2200 so XVim9Func
2201
2202 delete('XVim9Func')
2203enddef
2204
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002205let s:funcResult = 0
2206
2207def FuncNoArgNoRet()
Bram Moolenaar53900992020-08-22 19:02:02 +02002208 s:funcResult = 11
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002209enddef
2210
2211def FuncNoArgRetNumber(): number
Bram Moolenaar53900992020-08-22 19:02:02 +02002212 s:funcResult = 22
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002213 return 1234
2214enddef
2215
Bram Moolenaarec5929d2020-04-07 20:53:39 +02002216def FuncNoArgRetString(): string
Bram Moolenaar53900992020-08-22 19:02:02 +02002217 s:funcResult = 45
Bram Moolenaarec5929d2020-04-07 20:53:39 +02002218 return 'text'
2219enddef
2220
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002221def FuncOneArgNoRet(arg: number)
Bram Moolenaar53900992020-08-22 19:02:02 +02002222 s:funcResult = arg
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002223enddef
2224
2225def FuncOneArgRetNumber(arg: number): number
Bram Moolenaar53900992020-08-22 19:02:02 +02002226 s:funcResult = arg
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002227 return arg
2228enddef
2229
Bram Moolenaar08938ee2020-04-11 23:17:17 +02002230def FuncTwoArgNoRet(one: bool, two: number)
Bram Moolenaar53900992020-08-22 19:02:02 +02002231 s:funcResult = two
Bram Moolenaar08938ee2020-04-11 23:17:17 +02002232enddef
2233
Bram Moolenaar62aec932022-01-29 21:45:34 +00002234def s:FuncOneArgRetString(arg: string): string
Bram Moolenaarec5929d2020-04-07 20:53:39 +02002235 return arg
2236enddef
2237
Bram Moolenaar62aec932022-01-29 21:45:34 +00002238def s:FuncOneArgRetAny(arg: any): any
Bram Moolenaar89228602020-04-05 22:14:54 +02002239 return arg
2240enddef
2241
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002242def Test_func_type()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002243 var Ref1: func()
Bram Moolenaar53900992020-08-22 19:02:02 +02002244 s:funcResult = 0
Bram Moolenaar62aec932022-01-29 21:45:34 +00002245 Ref1 = g:FuncNoArgNoRet
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002246 Ref1()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002247 s:funcResult->assert_equal(11)
Bram Moolenaar4c683752020-04-05 21:38:23 +02002248
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002249 var Ref2: func
Bram Moolenaar53900992020-08-22 19:02:02 +02002250 s:funcResult = 0
Bram Moolenaar62aec932022-01-29 21:45:34 +00002251 Ref2 = g:FuncNoArgNoRet
Bram Moolenaar4c683752020-04-05 21:38:23 +02002252 Ref2()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002253 s:funcResult->assert_equal(11)
Bram Moolenaar4c683752020-04-05 21:38:23 +02002254
Bram Moolenaar53900992020-08-22 19:02:02 +02002255 s:funcResult = 0
Bram Moolenaar62aec932022-01-29 21:45:34 +00002256 Ref2 = g:FuncOneArgNoRet
Bram Moolenaar4c683752020-04-05 21:38:23 +02002257 Ref2(12)
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002258 s:funcResult->assert_equal(12)
Bram Moolenaar4c683752020-04-05 21:38:23 +02002259
Bram Moolenaar53900992020-08-22 19:02:02 +02002260 s:funcResult = 0
Bram Moolenaar62aec932022-01-29 21:45:34 +00002261 Ref2 = g:FuncNoArgRetNumber
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002262 Ref2()->assert_equal(1234)
2263 s:funcResult->assert_equal(22)
Bram Moolenaar4c683752020-04-05 21:38:23 +02002264
Bram Moolenaar53900992020-08-22 19:02:02 +02002265 s:funcResult = 0
Bram Moolenaar848fadd2022-01-30 15:28:30 +00002266 Ref2 = g:FuncOneArgRetNumber
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002267 Ref2(13)->assert_equal(13)
2268 s:funcResult->assert_equal(13)
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002269enddef
2270
Bram Moolenaar9978d472020-07-05 16:01:56 +02002271def Test_repeat_return_type()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002272 var res = 0
Bram Moolenaar9978d472020-07-05 16:01:56 +02002273 for n in repeat([1], 3)
2274 res += n
2275 endfor
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002276 res->assert_equal(3)
Bram Moolenaarfce82b32020-07-05 16:07:21 +02002277
2278 res = 0
2279 for n in add([1, 2], 3)
2280 res += n
2281 endfor
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002282 res->assert_equal(6)
Bram Moolenaar9978d472020-07-05 16:01:56 +02002283enddef
2284
Bram Moolenaar846178a2020-07-05 17:04:13 +02002285def Test_argv_return_type()
2286 next fileone filetwo
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002287 var res = ''
Bram Moolenaar846178a2020-07-05 17:04:13 +02002288 for name in argv()
2289 res ..= name
2290 endfor
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002291 res->assert_equal('fileonefiletwo')
Bram Moolenaar846178a2020-07-05 17:04:13 +02002292enddef
2293
Bram Moolenaarec5929d2020-04-07 20:53:39 +02002294def Test_func_type_part()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002295 var RefVoid: func: void
Bram Moolenaar62aec932022-01-29 21:45:34 +00002296 RefVoid = g:FuncNoArgNoRet
2297 RefVoid = g:FuncOneArgNoRet
2298 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 +00002299 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 +02002300
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002301 var RefAny: func(): any
Bram Moolenaar62aec932022-01-29 21:45:34 +00002302 RefAny = g:FuncNoArgRetNumber
Bram Moolenaar848fadd2022-01-30 15:28:30 +00002303 RefAny = g:FuncNoArgRetString
Bram Moolenaar62aec932022-01-29 21:45:34 +00002304 v9.CheckDefFailure(['var RefAny: func(): any', 'RefAny = g:FuncNoArgNoRet'], 'E1012: Type mismatch; expected func(): any but got func()')
2305 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 +02002306
Bram Moolenaar6abd3dc2020-10-04 14:17:32 +02002307 var RefAnyNoArgs: func: any = RefAny
2308
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002309 var RefNr: func: number
Bram Moolenaar62aec932022-01-29 21:45:34 +00002310 RefNr = g:FuncNoArgRetNumber
Bram Moolenaar848fadd2022-01-30 15:28:30 +00002311 RefNr = g:FuncOneArgRetNumber
Bram Moolenaar62aec932022-01-29 21:45:34 +00002312 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 +00002313 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 +02002314
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002315 var RefStr: func: string
Bram Moolenaar848fadd2022-01-30 15:28:30 +00002316 RefStr = g:FuncNoArgRetString
Bram Moolenaarec5929d2020-04-07 20:53:39 +02002317 RefStr = FuncOneArgRetString
Bram Moolenaar62aec932022-01-29 21:45:34 +00002318 v9.CheckDefFailure(['var RefStr: func: string', 'RefStr = g:FuncNoArgNoRet'], 'E1012: Type mismatch; expected func(...): string but got func()')
2319 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 +02002320enddef
2321
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002322def Test_func_type_fails()
Bram Moolenaar62aec932022-01-29 21:45:34 +00002323 v9.CheckDefFailure(['var ref1: func()'], 'E704:')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002324
Bram Moolenaar62aec932022-01-29 21:45:34 +00002325 v9.CheckDefFailure(['var Ref1: func()', 'Ref1 = g:FuncNoArgRetNumber'], 'E1012: Type mismatch; expected func() but got func(): number')
2326 v9.CheckDefFailure(['var Ref1: func()', 'Ref1 = g:FuncOneArgNoRet'], 'E1012: Type mismatch; expected func() but got func(number)')
Bram Moolenaar848fadd2022-01-30 15:28:30 +00002327 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 +00002328 v9.CheckDefFailure(['var Ref1: func(bool)', 'Ref1 = g:FuncTwoArgNoRet'], 'E1012: Type mismatch; expected func(bool) but got func(bool, number)')
2329 v9.CheckDefFailure(['var Ref1: func(?bool)', 'Ref1 = g:FuncTwoArgNoRet'], 'E1012: Type mismatch; expected func(?bool) but got func(bool, number)')
2330 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 +02002331
Bram Moolenaar62aec932022-01-29 21:45:34 +00002332 v9.CheckDefFailure(['var RefWrong: func(string ,number)'], 'E1068:')
2333 v9.CheckDefFailure(['var RefWrong: func(string,number)'], 'E1069:')
2334 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:')
2335 v9.CheckDefFailure(['var RefWrong: func(bool):string'], 'E1069:')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002336enddef
2337
Bram Moolenaar89228602020-04-05 22:14:54 +02002338def Test_func_return_type()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002339 var nr: number
Bram Moolenaar62aec932022-01-29 21:45:34 +00002340 nr = g:FuncNoArgRetNumber()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002341 nr->assert_equal(1234)
Bram Moolenaar89228602020-04-05 22:14:54 +02002342
2343 nr = FuncOneArgRetAny(122)
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002344 nr->assert_equal(122)
Bram Moolenaar89228602020-04-05 22:14:54 +02002345
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002346 var str: string
Bram Moolenaar89228602020-04-05 22:14:54 +02002347 str = FuncOneArgRetAny('yes')
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002348 str->assert_equal('yes')
Bram Moolenaar89228602020-04-05 22:14:54 +02002349
Bram Moolenaar62aec932022-01-29 21:45:34 +00002350 v9.CheckDefFailure(['var str: string', 'str = g:FuncNoArgRetNumber()'], 'E1012: Type mismatch; expected string but got number')
Bram Moolenaar89228602020-04-05 22:14:54 +02002351enddef
2352
Bram Moolenaar6abd3dc2020-10-04 14:17:32 +02002353def Test_func_common_type()
2354 def FuncOne(n: number): number
2355 return n
2356 enddef
2357 def FuncTwo(s: string): number
2358 return len(s)
2359 enddef
2360 def FuncThree(n: number, s: string): number
2361 return n + len(s)
2362 enddef
2363 var list = [FuncOne, FuncTwo, FuncThree]
2364 assert_equal(8, list[0](8))
2365 assert_equal(4, list[1]('word'))
2366 assert_equal(7, list[2](3, 'word'))
2367enddef
2368
Bram Moolenaar62aec932022-01-29 21:45:34 +00002369def s:MultiLine(
Bram Moolenaar5e774c72020-04-12 21:53:00 +02002370 arg1: string,
2371 arg2 = 1234,
2372 ...rest: list<string>
2373 ): string
2374 return arg1 .. arg2 .. join(rest, '-')
2375enddef
2376
Bram Moolenaar2c330432020-04-13 14:41:35 +02002377def MultiLineComment(
2378 arg1: string, # comment
2379 arg2 = 1234, # comment
2380 ...rest: list<string> # comment
2381 ): string # comment
2382 return arg1 .. arg2 .. join(rest, '-')
2383enddef
2384
Bram Moolenaar5e774c72020-04-12 21:53:00 +02002385def Test_multiline()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002386 MultiLine('text')->assert_equal('text1234')
2387 MultiLine('text', 777)->assert_equal('text777')
2388 MultiLine('text', 777, 'one')->assert_equal('text777one')
2389 MultiLine('text', 777, 'one', 'two')->assert_equal('text777one-two')
Bram Moolenaar5e774c72020-04-12 21:53:00 +02002390enddef
2391
Bram Moolenaar23e03252020-04-12 22:22:31 +02002392func Test_multiline_not_vim9()
Bram Moolenaar62aec932022-01-29 21:45:34 +00002393 call s:MultiLine('text')->assert_equal('text1234')
2394 call s:MultiLine('text', 777)->assert_equal('text777')
2395 call s:MultiLine('text', 777, 'one')->assert_equal('text777one')
2396 call s:MultiLine('text', 777, 'one', 'two')->assert_equal('text777one-two')
Bram Moolenaar23e03252020-04-12 22:22:31 +02002397endfunc
2398
Bram Moolenaar5e774c72020-04-12 21:53:00 +02002399
Bram Moolenaaree4e0c12020-04-06 21:35:05 +02002400" When using CheckScriptFailure() for the below test, E1010 is generated instead
2401" of E1056.
2402func Test_E1056_1059()
2403 let caught_1056 = 0
2404 try
2405 def F():
2406 return 1
2407 enddef
2408 catch /E1056:/
2409 let caught_1056 = 1
2410 endtry
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002411 eval caught_1056->assert_equal(1)
Bram Moolenaaree4e0c12020-04-06 21:35:05 +02002412
2413 let caught_1059 = 0
2414 try
2415 def F5(items : list)
2416 echo 'a'
2417 enddef
2418 catch /E1059:/
2419 let caught_1059 = 1
2420 endtry
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002421 eval caught_1059->assert_equal(1)
Bram Moolenaaree4e0c12020-04-06 21:35:05 +02002422endfunc
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002423
Bram Moolenaar015f4262020-05-05 21:25:22 +02002424func DelMe()
2425 echo 'DelMe'
2426endfunc
2427
Bram Moolenaarbf8feb52020-08-08 14:26:31 +02002428def Test_error_reporting()
2429 # comment lines at the start of the function
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002430 var lines =<< trim END
Bram Moolenaarbf8feb52020-08-08 14:26:31 +02002431 " comment
2432 def Func()
2433 # comment
2434 # comment
2435 invalid
2436 enddef
2437 defcompile
2438 END
Bram Moolenaar08052222020-09-14 17:04:31 +02002439 writefile(lines, 'Xdef')
Bram Moolenaarbf8feb52020-08-08 14:26:31 +02002440 try
2441 source Xdef
Bram Moolenaar7517ffd2020-08-14 18:35:07 +02002442 assert_report('should have failed')
Bram Moolenaarbf8feb52020-08-08 14:26:31 +02002443 catch /E476:/
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002444 v:exception->assert_match('Invalid command: invalid')
2445 v:throwpoint->assert_match(', line 3$')
Bram Moolenaarbf8feb52020-08-08 14:26:31 +02002446 endtry
Bram Moolenaar2d870f82020-12-05 13:41:01 +01002447 delfunc! g:Func
Bram Moolenaarbf8feb52020-08-08 14:26:31 +02002448
2449 # comment lines after the start of the function
2450 lines =<< trim END
2451 " comment
2452 def Func()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002453 var x = 1234
Bram Moolenaarbf8feb52020-08-08 14:26:31 +02002454 # comment
2455 # comment
2456 invalid
2457 enddef
2458 defcompile
2459 END
Bram Moolenaar08052222020-09-14 17:04:31 +02002460 writefile(lines, 'Xdef')
Bram Moolenaarbf8feb52020-08-08 14:26:31 +02002461 try
2462 source Xdef
Bram Moolenaar7517ffd2020-08-14 18:35:07 +02002463 assert_report('should have failed')
Bram Moolenaarbf8feb52020-08-08 14:26:31 +02002464 catch /E476:/
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002465 v:exception->assert_match('Invalid command: invalid')
2466 v:throwpoint->assert_match(', line 4$')
Bram Moolenaarbf8feb52020-08-08 14:26:31 +02002467 endtry
Bram Moolenaar2d870f82020-12-05 13:41:01 +01002468 delfunc! g:Func
Bram Moolenaarbf8feb52020-08-08 14:26:31 +02002469
Bram Moolenaar7517ffd2020-08-14 18:35:07 +02002470 lines =<< trim END
2471 vim9script
2472 def Func()
Bram Moolenaare0de1712020-12-02 17:36:54 +01002473 var db = {foo: 1, bar: 2}
Bram Moolenaar7517ffd2020-08-14 18:35:07 +02002474 # comment
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002475 var x = db.asdf
Bram Moolenaar7517ffd2020-08-14 18:35:07 +02002476 enddef
2477 defcompile
2478 Func()
2479 END
Bram Moolenaar08052222020-09-14 17:04:31 +02002480 writefile(lines, 'Xdef')
Bram Moolenaar7517ffd2020-08-14 18:35:07 +02002481 try
2482 source Xdef
2483 assert_report('should have failed')
2484 catch /E716:/
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002485 v:throwpoint->assert_match('_Func, line 3$')
Bram Moolenaar7517ffd2020-08-14 18:35:07 +02002486 endtry
Bram Moolenaar2d870f82020-12-05 13:41:01 +01002487 delfunc! g:Func
Bram Moolenaar7517ffd2020-08-14 18:35:07 +02002488
Bram Moolenaar08052222020-09-14 17:04:31 +02002489 delete('Xdef')
Bram Moolenaarbf8feb52020-08-08 14:26:31 +02002490enddef
2491
Bram Moolenaar015f4262020-05-05 21:25:22 +02002492def Test_deleted_function()
Bram Moolenaar62aec932022-01-29 21:45:34 +00002493 v9.CheckDefExecFailure([
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002494 'var RefMe: func = function("g:DelMe")',
Bram Moolenaar015f4262020-05-05 21:25:22 +02002495 'delfunc g:DelMe',
2496 'echo RefMe()'], 'E117:')
2497enddef
2498
2499def Test_unknown_function()
Bram Moolenaar62aec932022-01-29 21:45:34 +00002500 v9.CheckDefExecFailure([
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002501 'var Ref: func = function("NotExist")',
Bram Moolenaar9b7bf9e2020-07-11 22:14:59 +02002502 'delfunc g:NotExist'], 'E700:')
Bram Moolenaar015f4262020-05-05 21:25:22 +02002503enddef
2504
Bram Moolenaar62aec932022-01-29 21:45:34 +00002505def s:RefFunc(Ref: func(any): any): string
Bram Moolenaarc8cd2b32020-05-01 19:29:08 +02002506 return Ref('more')
2507enddef
2508
2509def Test_closure_simple()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002510 var local = 'some '
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01002511 RefFunc((s) => local .. s)->assert_equal('some more')
Bram Moolenaarc8cd2b32020-05-01 19:29:08 +02002512enddef
2513
Bram Moolenaar62aec932022-01-29 21:45:34 +00002514def s:MakeRef()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002515 var local = 'some '
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01002516 g:Ref = (s) => local .. s
Bram Moolenaarbf67ea12020-05-02 17:52:42 +02002517enddef
2518
2519def Test_closure_ref_after_return()
2520 MakeRef()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002521 g:Ref('thing')->assert_equal('some thing')
Bram Moolenaarbf67ea12020-05-02 17:52:42 +02002522 unlet g:Ref
2523enddef
2524
Bram Moolenaar62aec932022-01-29 21:45:34 +00002525def s:MakeTwoRefs()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002526 var local = ['some']
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01002527 g:Extend = (s) => local->add(s)
2528 g:Read = () => local
Bram Moolenaar5adc55c2020-05-02 23:12:58 +02002529enddef
2530
2531def Test_closure_two_refs()
2532 MakeTwoRefs()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002533 join(g:Read(), ' ')->assert_equal('some')
Bram Moolenaar5adc55c2020-05-02 23:12:58 +02002534 g:Extend('more')
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002535 join(g:Read(), ' ')->assert_equal('some more')
Bram Moolenaar5adc55c2020-05-02 23:12:58 +02002536 g:Extend('even')
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002537 join(g:Read(), ' ')->assert_equal('some more even')
Bram Moolenaar5adc55c2020-05-02 23:12:58 +02002538
2539 unlet g:Extend
2540 unlet g:Read
2541enddef
2542
Bram Moolenaar62aec932022-01-29 21:45:34 +00002543def s:ReadRef(Ref: func(): list<string>): string
Bram Moolenaar5adc55c2020-05-02 23:12:58 +02002544 return join(Ref(), ' ')
2545enddef
2546
Bram Moolenaar62aec932022-01-29 21:45:34 +00002547def s:ExtendRef(Ref: func(string): list<string>, add: string)
Bram Moolenaar5adc55c2020-05-02 23:12:58 +02002548 Ref(add)
2549enddef
2550
2551def Test_closure_two_indirect_refs()
Bram Moolenaarf7779c62020-05-03 15:38:16 +02002552 MakeTwoRefs()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002553 ReadRef(g:Read)->assert_equal('some')
Bram Moolenaar5adc55c2020-05-02 23:12:58 +02002554 ExtendRef(g:Extend, 'more')
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002555 ReadRef(g:Read)->assert_equal('some more')
Bram Moolenaar5adc55c2020-05-02 23:12:58 +02002556 ExtendRef(g:Extend, 'even')
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002557 ReadRef(g:Read)->assert_equal('some more even')
Bram Moolenaar5adc55c2020-05-02 23:12:58 +02002558
2559 unlet g:Extend
2560 unlet g:Read
2561enddef
Bram Moolenaarbf67ea12020-05-02 17:52:42 +02002562
Bram Moolenaar62aec932022-01-29 21:45:34 +00002563def s:MakeArgRefs(theArg: string)
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002564 var local = 'loc_val'
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01002565 g:UseArg = (s) => theArg .. '/' .. local .. '/' .. s
Bram Moolenaar2fd4cd72020-05-03 22:30:49 +02002566enddef
2567
Bram Moolenaar62aec932022-01-29 21:45:34 +00002568def s:MakeArgRefsVarargs(theArg: string, ...rest: list<string>)
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002569 var local = 'the_loc'
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01002570 g:UseVararg = (s) => theArg .. '/' .. local .. '/' .. s .. '/' .. join(rest)
Bram Moolenaar2fd4cd72020-05-03 22:30:49 +02002571enddef
2572
2573def Test_closure_using_argument()
2574 MakeArgRefs('arg_val')
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002575 g:UseArg('call_val')->assert_equal('arg_val/loc_val/call_val')
Bram Moolenaar2fd4cd72020-05-03 22:30:49 +02002576
2577 MakeArgRefsVarargs('arg_val', 'one', 'two')
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002578 g:UseVararg('call_val')->assert_equal('arg_val/the_loc/call_val/one two')
Bram Moolenaar2fd4cd72020-05-03 22:30:49 +02002579
2580 unlet g:UseArg
2581 unlet g:UseVararg
Bram Moolenaar44ec21c2021-02-12 21:50:57 +01002582
2583 var lines =<< trim END
2584 vim9script
2585 def Test(Fun: func(number): number): list<number>
2586 return map([1, 2, 3], (_, i) => Fun(i))
2587 enddef
2588 def Inc(nr: number): number
2589 return nr + 2
2590 enddef
2591 assert_equal([3, 4, 5], Test(Inc))
2592 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002593 v9.CheckScriptSuccess(lines)
Bram Moolenaar2fd4cd72020-05-03 22:30:49 +02002594enddef
2595
Bram Moolenaar62aec932022-01-29 21:45:34 +00002596def s:MakeGetAndAppendRefs()
Bram Moolenaar85d5e2b2020-10-10 14:13:01 +02002597 var local = 'a'
2598
2599 def Append(arg: string)
2600 local ..= arg
2601 enddef
2602 g:Append = Append
2603
2604 def Get(): string
2605 return local
2606 enddef
2607 g:Get = Get
2608enddef
2609
2610def Test_closure_append_get()
2611 MakeGetAndAppendRefs()
2612 g:Get()->assert_equal('a')
2613 g:Append('-b')
2614 g:Get()->assert_equal('a-b')
2615 g:Append('-c')
2616 g:Get()->assert_equal('a-b-c')
2617
2618 unlet g:Append
2619 unlet g:Get
2620enddef
Bram Moolenaarb68b3462020-05-06 21:06:30 +02002621
Bram Moolenaar04b12692020-05-04 23:24:44 +02002622def Test_nested_closure()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002623 var local = 'text'
Bram Moolenaar04b12692020-05-04 23:24:44 +02002624 def Closure(arg: string): string
2625 return local .. arg
2626 enddef
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002627 Closure('!!!')->assert_equal('text!!!')
Bram Moolenaar04b12692020-05-04 23:24:44 +02002628enddef
2629
Bram Moolenaar62aec932022-01-29 21:45:34 +00002630func s:GetResult(Ref)
Bram Moolenaar6f5b6df2020-05-16 21:20:12 +02002631 return a:Ref('some')
2632endfunc
2633
2634def Test_call_closure_not_compiled()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002635 var text = 'text'
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01002636 g:Ref = (s) => s .. text
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002637 GetResult(g:Ref)->assert_equal('sometext')
Bram Moolenaar6f5b6df2020-05-16 21:20:12 +02002638enddef
2639
Bram Moolenaar7cbfaa52020-09-18 21:25:32 +02002640def Test_double_closure_fails()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002641 var lines =<< trim END
Bram Moolenaar7cbfaa52020-09-18 21:25:32 +02002642 vim9script
2643 def Func()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002644 var name = 0
2645 for i in range(2)
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01002646 timer_start(0, () => name)
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002647 endfor
Bram Moolenaar7cbfaa52020-09-18 21:25:32 +02002648 enddef
2649 Func()
2650 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002651 v9.CheckScriptSuccess(lines)
Bram Moolenaar7cbfaa52020-09-18 21:25:32 +02002652enddef
2653
Bram Moolenaar85d5e2b2020-10-10 14:13:01 +02002654def Test_nested_closure_used()
2655 var lines =<< trim END
2656 vim9script
2657 def Func()
2658 var x = 'hello'
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01002659 var Closure = () => x
2660 g:Myclosure = () => Closure()
Bram Moolenaar85d5e2b2020-10-10 14:13:01 +02002661 enddef
2662 Func()
2663 assert_equal('hello', g:Myclosure())
2664 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002665 v9.CheckScriptSuccess(lines)
Bram Moolenaar85d5e2b2020-10-10 14:13:01 +02002666enddef
Bram Moolenaar0876c782020-10-07 19:08:04 +02002667
Bram Moolenaarc70bdab2020-09-26 19:59:38 +02002668def Test_nested_closure_fails()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002669 var lines =<< trim END
Bram Moolenaarc70bdab2020-09-26 19:59:38 +02002670 vim9script
2671 def FuncA()
2672 FuncB(0)
2673 enddef
2674 def FuncB(n: number): list<string>
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01002675 return map([0], (_, v) => n)
Bram Moolenaarc70bdab2020-09-26 19:59:38 +02002676 enddef
2677 FuncA()
2678 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002679 v9.CheckScriptFailure(lines, 'E1012:')
Bram Moolenaarc70bdab2020-09-26 19:59:38 +02002680enddef
2681
Bram Moolenaarf112f302020-12-20 17:47:52 +01002682def Test_global_closure()
2683 var lines =<< trim END
2684 vim9script
2685 def ReverseEveryNLines(n: number, line1: number, line2: number)
2686 var mods = 'sil keepj keepp lockm '
2687 var range = ':' .. line1 .. ',' .. line2
2688 def g:Offset(): number
2689 var offset = (line('.') - line1 + 1) % n
2690 return offset != 0 ? offset : n
2691 enddef
2692 exe mods .. range .. 'g/^/exe "m .-" .. g:Offset()'
2693 enddef
2694
2695 new
2696 repeat(['aaa', 'bbb', 'ccc'], 3)->setline(1)
2697 ReverseEveryNLines(3, 1, 9)
2698 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002699 v9.CheckScriptSuccess(lines)
Bram Moolenaarf112f302020-12-20 17:47:52 +01002700 var expected = repeat(['ccc', 'bbb', 'aaa'], 3)
2701 assert_equal(expected, getline(1, 9))
2702 bwipe!
2703enddef
2704
Bram Moolenaarcd45ed02020-12-22 17:35:54 +01002705def Test_global_closure_called_directly()
2706 var lines =<< trim END
2707 vim9script
2708 def Outer()
2709 var x = 1
2710 def g:Inner()
2711 var y = x
2712 x += 1
2713 assert_equal(1, y)
2714 enddef
2715 g:Inner()
2716 assert_equal(2, x)
2717 enddef
2718 Outer()
2719 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002720 v9.CheckScriptSuccess(lines)
Bram Moolenaarcd45ed02020-12-22 17:35:54 +01002721 delfunc g:Inner
2722enddef
2723
Bram Moolenaar69c76172021-12-02 16:38:52 +00002724def Test_closure_called_from_legacy()
2725 var lines =<< trim END
2726 vim9script
2727 def Func()
2728 var outer = 'foo'
2729 var F = () => {
2730 outer = 'bar'
2731 }
2732 execute printf('call %s()', string(F))
2733 enddef
2734 Func()
2735 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002736 v9.CheckScriptFailure(lines, 'E1248')
Bram Moolenaar69c76172021-12-02 16:38:52 +00002737enddef
2738
Bram Moolenaar34c54eb2020-11-25 19:15:19 +01002739def Test_failure_in_called_function()
2740 # this was using the frame index as the return value
2741 var lines =<< trim END
2742 vim9script
2743 au TerminalWinOpen * eval [][0]
2744 def PopupTerm(a: any)
2745 # make sure typvals on stack are string
2746 ['a', 'b', 'c', 'd', 'e', 'f', 'g']->join()
2747 FireEvent()
2748 enddef
2749 def FireEvent()
2750 do TerminalWinOpen
2751 enddef
2752 # use try/catch to make eval fail
2753 try
2754 call PopupTerm(0)
2755 catch
2756 endtry
2757 au! TerminalWinOpen
2758 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002759 v9.CheckScriptSuccess(lines)
Bram Moolenaar34c54eb2020-11-25 19:15:19 +01002760enddef
2761
Bram Moolenaar5366e1a2020-10-01 13:01:34 +02002762def Test_nested_lambda()
2763 var lines =<< trim END
2764 vim9script
2765 def Func()
2766 var x = 4
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01002767 var Lambda1 = () => 7
2768 var Lambda2 = () => [Lambda1(), x]
Bram Moolenaar5366e1a2020-10-01 13:01:34 +02002769 var res = Lambda2()
2770 assert_equal([7, 4], res)
2771 enddef
2772 Func()
2773 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002774 v9.CheckScriptSuccess(lines)
Bram Moolenaar5366e1a2020-10-01 13:01:34 +02002775enddef
2776
Bram Moolenaarc04f2a42021-06-09 19:30:03 +02002777def Test_double_nested_lambda()
2778 var lines =<< trim END
2779 vim9script
2780 def F(head: string): func(string): func(string): string
2781 return (sep: string): func(string): string => ((tail: string): string => {
2782 return head .. sep .. tail
2783 })
2784 enddef
2785 assert_equal('hello-there', F('hello')('-')('there'))
2786 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002787 v9.CheckScriptSuccess(lines)
Bram Moolenaarc04f2a42021-06-09 19:30:03 +02002788enddef
2789
Bram Moolenaar074f84c2021-05-18 11:47:44 +02002790def Test_nested_inline_lambda()
Bram Moolenaar074f84c2021-05-18 11:47:44 +02002791 var lines =<< trim END
2792 vim9script
2793 def F(text: string): func(string): func(string): string
2794 return (arg: string): func(string): string => ((sep: string): string => {
Bram Moolenaar23e2e112021-08-03 21:16:18 +02002795 return sep .. arg .. text
Bram Moolenaar074f84c2021-05-18 11:47:44 +02002796 })
2797 enddef
Bram Moolenaar23e2e112021-08-03 21:16:18 +02002798 assert_equal('--there++', F('++')('there')('--'))
Bram Moolenaar074f84c2021-05-18 11:47:44 +02002799 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002800 v9.CheckScriptSuccess(lines)
Bram Moolenaar5245beb2021-07-15 22:03:50 +02002801
2802 lines =<< trim END
2803 vim9script
2804 echo range(4)->mapnew((_, v) => {
2805 return range(v) ->mapnew((_, s) => {
2806 return string(s)
2807 })
2808 })
2809 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002810 v9.CheckScriptSuccess(lines)
Bram Moolenaarc6ba2f92021-07-18 13:42:29 +02002811
2812 lines =<< trim END
2813 vim9script
2814
Bram Moolenaar3787f262022-02-07 21:54:01 +00002815 def s:Func()
Bram Moolenaarc6ba2f92021-07-18 13:42:29 +02002816 range(10)
2817 ->mapnew((_, _) => ({
2818 key: range(10)->mapnew((_, _) => {
2819 return ' '
2820 }),
2821 }))
2822 enddef
2823
2824 defcomp
2825 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002826 v9.CheckScriptSuccess(lines)
Bram Moolenaar074f84c2021-05-18 11:47:44 +02002827enddef
2828
Bram Moolenaar52bf81c2020-11-17 18:50:44 +01002829def Shadowed(): list<number>
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01002830 var FuncList: list<func: number> = [() => 42]
Bram Moolenaar75ab91f2021-01-10 22:42:50 +01002831 return FuncList->mapnew((_, Shadowed) => Shadowed())
Bram Moolenaar52bf81c2020-11-17 18:50:44 +01002832enddef
2833
2834def Test_lambda_arg_shadows_func()
Bram Moolenaar62aec932022-01-29 21:45:34 +00002835 assert_equal([42], g:Shadowed())
Bram Moolenaar52bf81c2020-11-17 18:50:44 +01002836enddef
2837
Bram Moolenaar62aec932022-01-29 21:45:34 +00002838def s:Line_continuation_in_def(dir: string = ''): string
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002839 var path: string = empty(dir)
2840 \ ? 'empty'
2841 \ : 'full'
2842 return path
Bram Moolenaaracd4c5e2020-06-22 19:39:03 +02002843enddef
2844
2845def Test_line_continuation_in_def()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002846 Line_continuation_in_def('.')->assert_equal('full')
Bram Moolenaaracd4c5e2020-06-22 19:39:03 +02002847enddef
2848
Bram Moolenaar2ea95b62020-11-19 21:47:56 +01002849def Test_script_var_in_lambda()
2850 var lines =<< trim END
2851 vim9script
2852 var script = 'test'
Bram Moolenaarbb8a7ce2021-04-10 20:10:26 +02002853 assert_equal(['test'], map(['one'], (_, _) => script))
Bram Moolenaar2ea95b62020-11-19 21:47:56 +01002854 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002855 v9.CheckScriptSuccess(lines)
Bram Moolenaar2ea95b62020-11-19 21:47:56 +01002856enddef
2857
Bram Moolenaar62aec932022-01-29 21:45:34 +00002858def s:Line_continuation_in_lambda(): list<string>
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002859 var x = range(97, 100)
Bram Moolenaar75ab91f2021-01-10 22:42:50 +01002860 ->mapnew((_, v) => nr2char(v)
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01002861 ->toupper())
Bram Moolenaar7a4b8982020-07-08 17:36:21 +02002862 ->reverse()
2863 return x
2864enddef
2865
2866def Test_line_continuation_in_lambda()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002867 Line_continuation_in_lambda()->assert_equal(['D', 'C', 'B', 'A'])
Bram Moolenaarf898f7c2021-01-16 18:09:52 +01002868
2869 var lines =<< trim END
2870 vim9script
2871 var res = [{n: 1, m: 2, s: 'xxx'}]
2872 ->mapnew((_, v: dict<any>): string => printf('%d:%d:%s',
2873 v.n,
2874 v.m,
2875 substitute(v.s, '.*', 'yyy', '')
2876 ))
2877 assert_equal(['1:2:yyy'], res)
2878 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002879 v9.CheckScriptSuccess(lines)
Bram Moolenaar7a4b8982020-07-08 17:36:21 +02002880enddef
2881
Bram Moolenaarb6571982021-01-08 22:24:19 +01002882def Test_list_lambda()
2883 timer_start(1000, (_) => 0)
2884 var body = execute(timer_info()[0].callback
2885 ->string()
2886 ->substitute("('", ' ', '')
2887 ->substitute("')", '', '')
2888 ->substitute('function\zs', ' ', ''))
Bram Moolenaar767034c2021-04-09 17:24:52 +02002889 assert_match('def <lambda>\d\+(_: any): number\n1 return 0\n enddef', body)
Bram Moolenaarb6571982021-01-08 22:24:19 +01002890enddef
2891
Bram Moolenaar3c77b6a2021-07-25 18:07:00 +02002892def Test_lambda_block_variable()
Bram Moolenaar88421d62021-07-24 14:14:52 +02002893 var lines =<< trim END
2894 vim9script
2895 var flist: list<func>
2896 for i in range(10)
2897 var inloop = i
2898 flist[i] = () => inloop
2899 endfor
2900 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002901 v9.CheckScriptSuccess(lines)
Bram Moolenaar88421d62021-07-24 14:14:52 +02002902
2903 lines =<< trim END
2904 vim9script
2905 if true
2906 var outloop = 5
2907 var flist: list<func>
2908 for i in range(10)
2909 flist[i] = () => outloop
2910 endfor
2911 endif
2912 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002913 v9.CheckScriptSuccess(lines)
Bram Moolenaar88421d62021-07-24 14:14:52 +02002914
2915 lines =<< trim END
2916 vim9script
2917 if true
2918 var outloop = 5
2919 endif
2920 var flist: list<func>
2921 for i in range(10)
2922 flist[i] = () => outloop
2923 endfor
2924 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002925 v9.CheckScriptFailure(lines, 'E1001: Variable not found: outloop', 1)
Bram Moolenaar3c77b6a2021-07-25 18:07:00 +02002926
2927 lines =<< trim END
2928 vim9script
2929 for i in range(10)
2930 var Ref = () => 0
2931 endfor
2932 assert_equal(0, ((i) => 0)(0))
2933 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002934 v9.CheckScriptSuccess(lines)
Bram Moolenaar88421d62021-07-24 14:14:52 +02002935enddef
2936
Bram Moolenaar96cf4ba2021-04-24 14:15:41 +02002937def Test_legacy_lambda()
2938 legacy echo {x -> 'hello ' .. x}('foo')
Bram Moolenaardc4c2302021-04-25 13:54:42 +02002939
Bram Moolenaar96cf4ba2021-04-24 14:15:41 +02002940 var lines =<< trim END
2941 echo {x -> 'hello ' .. x}('foo')
2942 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002943 v9.CheckDefAndScriptFailure(lines, 'E720:')
Bram Moolenaardc4c2302021-04-25 13:54:42 +02002944
2945 lines =<< trim END
2946 vim9script
2947 def Func()
2948 echo (() => 'no error')()
2949 enddef
2950 legacy call s:Func()
2951 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002952 v9.CheckScriptSuccess(lines)
Bram Moolenaar96cf4ba2021-04-24 14:15:41 +02002953enddef
2954
Bram Moolenaarce024c32021-06-26 13:00:49 +02002955def Test_legacy()
2956 var lines =<< trim END
2957 vim9script
2958 func g:LegacyFunction()
2959 let g:legacyvar = 1
2960 endfunc
2961 def Testit()
2962 legacy call g:LegacyFunction()
2963 enddef
2964 Testit()
2965 assert_equal(1, g:legacyvar)
2966 unlet g:legacyvar
2967 delfunc g:LegacyFunction
2968 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00002969 v9.CheckScriptSuccess(lines)
Bram Moolenaarce024c32021-06-26 13:00:49 +02002970enddef
2971
Bram Moolenaarc3cb1c92021-06-02 16:47:53 +02002972def Test_legacy_errors()
2973 for cmd in ['if', 'elseif', 'else', 'endif',
2974 'for', 'endfor', 'continue', 'break',
2975 'while', 'endwhile',
2976 'try', 'catch', 'finally', 'endtry']
Bram Moolenaar62aec932022-01-29 21:45:34 +00002977 v9.CheckDefFailure(['legacy ' .. cmd .. ' expr'], 'E1189:')
Bram Moolenaarc3cb1c92021-06-02 16:47:53 +02002978 endfor
2979enddef
2980
Bram Moolenaarb1b6f4d2021-09-13 18:25:54 +02002981def Test_call_legacy_with_dict()
2982 var lines =<< trim END
2983 vim9script
2984 func Legacy() dict
2985 let g:result = self.value
2986 endfunc
2987 def TestDirect()
2988 var d = {value: 'yes', func: Legacy}
2989 d.func()
2990 enddef
2991 TestDirect()
2992 assert_equal('yes', g:result)
2993 unlet g:result
2994
2995 def TestIndirect()
2996 var d = {value: 'foo', func: Legacy}
2997 var Fi = d.func
2998 Fi()
2999 enddef
3000 TestIndirect()
3001 assert_equal('foo', g:result)
3002 unlet g:result
3003
3004 var d = {value: 'bar', func: Legacy}
3005 d.func()
3006 assert_equal('bar', g:result)
3007 unlet g:result
3008 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003009 v9.CheckScriptSuccess(lines)
Bram Moolenaarb1b6f4d2021-09-13 18:25:54 +02003010enddef
3011
Bram Moolenaar62aec932022-01-29 21:45:34 +00003012def s:DoFilterThis(a: string): list<string>
Bram Moolenaarab360522021-01-10 14:02:28 +01003013 # closure nested inside another closure using argument
3014 var Filter = (l) => filter(l, (_, v) => stridx(v, a) == 0)
3015 return ['x', 'y', 'a', 'x2', 'c']->Filter()
3016enddef
3017
3018def Test_nested_closure_using_argument()
3019 assert_equal(['x', 'x2'], DoFilterThis('x'))
3020enddef
3021
Bram Moolenaar0186e582021-01-10 18:33:11 +01003022def Test_triple_nested_closure()
3023 var what = 'x'
3024 var Match = (val: string, cmp: string): bool => stridx(val, cmp) == 0
3025 var Filter = (l) => filter(l, (_, v) => Match(v, what))
3026 assert_equal(['x', 'x2'], ['x', 'y', 'a', 'x2', 'c']->Filter())
3027enddef
3028
Bram Moolenaar8f510af2020-07-05 18:48:23 +02003029func Test_silent_echo()
Bram Moolenaar47e7d702020-07-05 18:18:42 +02003030 CheckScreendump
Bram Moolenaar3b309f12021-12-13 18:19:55 +00003031 call Run_Test_silent_echo()
3032endfunc
Bram Moolenaar47e7d702020-07-05 18:18:42 +02003033
Bram Moolenaar3b309f12021-12-13 18:19:55 +00003034def Run_Test_silent_echo()
3035 var lines =<< trim END
Bram Moolenaar47e7d702020-07-05 18:18:42 +02003036 vim9script
3037 def EchoNothing()
3038 silent echo ''
3039 enddef
3040 defcompile
3041 END
Bram Moolenaar3b309f12021-12-13 18:19:55 +00003042 writefile(lines, 'XTest_silent_echo')
Bram Moolenaar47e7d702020-07-05 18:18:42 +02003043
Bram Moolenaar3b309f12021-12-13 18:19:55 +00003044 # Check that the balloon shows up after a mouse move
Bram Moolenaar62aec932022-01-29 21:45:34 +00003045 var buf = g:RunVimInTerminal('-S XTest_silent_echo', {'rows': 6})
Bram Moolenaar3b309f12021-12-13 18:19:55 +00003046 term_sendkeys(buf, ":abc")
Bram Moolenaar62aec932022-01-29 21:45:34 +00003047 g:VerifyScreenDump(buf, 'Test_vim9_silent_echo', {})
Bram Moolenaar47e7d702020-07-05 18:18:42 +02003048
Bram Moolenaar3b309f12021-12-13 18:19:55 +00003049 # clean up
Bram Moolenaar62aec932022-01-29 21:45:34 +00003050 g:StopVimInTerminal(buf)
Bram Moolenaar3b309f12021-12-13 18:19:55 +00003051 delete('XTest_silent_echo')
3052enddef
Bram Moolenaar47e7d702020-07-05 18:18:42 +02003053
Bram Moolenaar171fb922020-10-28 16:54:47 +01003054def SilentlyError()
3055 execute('silent! invalid')
3056 g:did_it = 'yes'
3057enddef
3058
Bram Moolenaar62aec932022-01-29 21:45:34 +00003059func s:UserError()
Bram Moolenaar28ee8922020-10-28 20:20:00 +01003060 silent! invalid
3061endfunc
3062
3063def SilentlyUserError()
3064 UserError()
3065 g:did_it = 'yes'
3066enddef
Bram Moolenaar171fb922020-10-28 16:54:47 +01003067
3068" This can't be a :def function, because the assert would not be reached.
Bram Moolenaar171fb922020-10-28 16:54:47 +01003069func Test_ignore_silent_error()
3070 let g:did_it = 'no'
3071 call SilentlyError()
3072 call assert_equal('yes', g:did_it)
3073
Bram Moolenaar28ee8922020-10-28 20:20:00 +01003074 let g:did_it = 'no'
3075 call SilentlyUserError()
3076 call assert_equal('yes', g:did_it)
Bram Moolenaar171fb922020-10-28 16:54:47 +01003077
3078 unlet g:did_it
3079endfunc
3080
Bram Moolenaarcd030c42020-10-30 21:49:40 +01003081def Test_ignore_silent_error_in_filter()
3082 var lines =<< trim END
3083 vim9script
3084 def Filter(winid: number, key: string): bool
3085 if key == 'o'
3086 silent! eval [][0]
3087 return true
3088 endif
3089 return popup_filter_menu(winid, key)
3090 enddef
3091
Bram Moolenaare0de1712020-12-02 17:36:54 +01003092 popup_create('popup', {filter: Filter})
Bram Moolenaarcd030c42020-10-30 21:49:40 +01003093 feedkeys("o\r", 'xnt')
3094 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003095 v9.CheckScriptSuccess(lines)
Bram Moolenaarcd030c42020-10-30 21:49:40 +01003096enddef
3097
Bram Moolenaar62aec932022-01-29 21:45:34 +00003098def s:Fibonacci(n: number): number
Bram Moolenaar4b9bd692020-09-05 21:57:53 +02003099 if n < 2
3100 return n
3101 else
3102 return Fibonacci(n - 1) + Fibonacci(n - 2)
3103 endif
3104enddef
3105
Bram Moolenaar985116a2020-07-12 17:31:09 +02003106def Test_recursive_call()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02003107 Fibonacci(20)->assert_equal(6765)
Bram Moolenaar985116a2020-07-12 17:31:09 +02003108enddef
3109
Bram Moolenaar62aec932022-01-29 21:45:34 +00003110def s:TreeWalk(dir: string): list<any>
Bram Moolenaar75ab91f2021-01-10 22:42:50 +01003111 return readdir(dir)->mapnew((_, val) =>
Bram Moolenaar08f7a412020-07-13 20:41:08 +02003112 fnamemodify(dir .. '/' .. val, ':p')->isdirectory()
Bram Moolenaar2bede172020-11-19 18:53:18 +01003113 ? {[val]: TreeWalk(dir .. '/' .. val)}
Bram Moolenaar08f7a412020-07-13 20:41:08 +02003114 : val
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01003115 )
Bram Moolenaar08f7a412020-07-13 20:41:08 +02003116enddef
3117
3118def Test_closure_in_map()
3119 mkdir('XclosureDir/tdir', 'p')
3120 writefile(['111'], 'XclosureDir/file1')
3121 writefile(['222'], 'XclosureDir/file2')
3122 writefile(['333'], 'XclosureDir/tdir/file3')
3123
Bram Moolenaare0de1712020-12-02 17:36:54 +01003124 TreeWalk('XclosureDir')->assert_equal(['file1', 'file2', {tdir: ['file3']}])
Bram Moolenaar08f7a412020-07-13 20:41:08 +02003125
3126 delete('XclosureDir', 'rf')
3127enddef
3128
Bram Moolenaar7b5d5442020-10-04 13:42:34 +02003129def Test_invalid_function_name()
3130 var lines =<< trim END
3131 vim9script
3132 def s: list<string>
3133 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003134 v9.CheckScriptFailure(lines, 'E129:')
Bram Moolenaar7b5d5442020-10-04 13:42:34 +02003135
3136 lines =<< trim END
3137 vim9script
3138 def g: list<string>
3139 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003140 v9.CheckScriptFailure(lines, 'E129:')
Bram Moolenaar7b5d5442020-10-04 13:42:34 +02003141
3142 lines =<< trim END
3143 vim9script
3144 def <SID>: list<string>
3145 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003146 v9.CheckScriptFailure(lines, 'E884:')
Bram Moolenaar7b5d5442020-10-04 13:42:34 +02003147
3148 lines =<< trim END
3149 vim9script
3150 def F list<string>
3151 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003152 v9.CheckScriptFailure(lines, 'E488:')
Bram Moolenaar7b5d5442020-10-04 13:42:34 +02003153enddef
3154
Bram Moolenaara90afb92020-07-15 22:38:56 +02003155def Test_partial_call()
Bram Moolenaarf78da4f2021-08-01 15:40:31 +02003156 var lines =<< trim END
3157 var Xsetlist: func
3158 Xsetlist = function('setloclist', [0])
3159 Xsetlist([], ' ', {title: 'test'})
3160 getloclist(0, {title: 1})->assert_equal({title: 'test'})
Bram Moolenaara90afb92020-07-15 22:38:56 +02003161
Bram Moolenaarf78da4f2021-08-01 15:40:31 +02003162 Xsetlist = function('setloclist', [0, [], ' '])
3163 Xsetlist({title: 'test'})
3164 getloclist(0, {title: 1})->assert_equal({title: 'test'})
Bram Moolenaara90afb92020-07-15 22:38:56 +02003165
Bram Moolenaarf78da4f2021-08-01 15:40:31 +02003166 Xsetlist = function('setqflist')
3167 Xsetlist([], ' ', {title: 'test'})
3168 getqflist({title: 1})->assert_equal({title: 'test'})
Bram Moolenaara90afb92020-07-15 22:38:56 +02003169
Bram Moolenaarf78da4f2021-08-01 15:40:31 +02003170 Xsetlist = function('setqflist', [[], ' '])
3171 Xsetlist({title: 'test'})
3172 getqflist({title: 1})->assert_equal({title: 'test'})
Bram Moolenaar6abd3dc2020-10-04 14:17:32 +02003173
Bram Moolenaarf78da4f2021-08-01 15:40:31 +02003174 var Len: func: number = function('len', ['word'])
3175 assert_equal(4, Len())
3176
3177 var RepeatFunc = function('repeat', ['o'])
3178 assert_equal('ooooo', RepeatFunc(5))
3179 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003180 v9.CheckDefAndScriptSuccess(lines)
Bram Moolenaarc66f6452021-08-19 21:08:30 +02003181
3182 lines =<< trim END
3183 vim9script
3184 def Foo(Parser: any)
3185 enddef
3186 var Expr: func(dict<any>): dict<any>
3187 const Call = Foo(Expr)
3188 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003189 v9.CheckScriptFailure(lines, 'E1235:')
Bram Moolenaara90afb92020-07-15 22:38:56 +02003190enddef
3191
Bram Moolenaarfe1bfc92022-02-06 13:55:03 +00003192" Using "idx" from a legacy global function does not work.
3193" This caused a crash when called from legacy context.
3194func Test_partial_call_fails()
3195 let lines =<< trim END
3196 vim9script
3197
3198 var l = ['a', 'b', 'c']
3199 def Iter(container: any): any
3200 var idx = -1
3201 var obj = {state: container}
Bram Moolenaarf681cfb2022-02-07 20:30:57 +00003202 def g:NextItem__(self: dict<any>): any
Bram Moolenaarfe1bfc92022-02-06 13:55:03 +00003203 ++idx
3204 return self.state[idx]
3205 enddef
Bram Moolenaarf681cfb2022-02-07 20:30:57 +00003206 obj.__next__ = function('g:NextItem__', [obj])
Bram Moolenaarfe1bfc92022-02-06 13:55:03 +00003207 return obj
3208 enddef
3209
3210 var it = Iter(l)
3211 echo it.__next__()
3212 END
3213 call writefile(lines, 'XpartialCall')
3214 try
3215 source XpartialCall
3216 catch /E1248:/
3217 endtry
3218 call delete('XpartialCall')
3219endfunc
3220
Bram Moolenaar2dd0a2c2020-08-08 15:10:27 +02003221def Test_cmd_modifier()
3222 tab echo '0'
Bram Moolenaar62aec932022-01-29 21:45:34 +00003223 v9.CheckDefFailure(['5tab echo 3'], 'E16:')
Bram Moolenaar2dd0a2c2020-08-08 15:10:27 +02003224enddef
3225
3226def Test_restore_modifiers()
3227 # check that when compiling a :def function command modifiers are not messed
3228 # up.
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02003229 var lines =<< trim END
Bram Moolenaar2dd0a2c2020-08-08 15:10:27 +02003230 vim9script
3231 set eventignore=
3232 autocmd QuickFixCmdPost * copen
3233 def AutocmdsDisabled()
Bram Moolenaarc3235272021-07-10 19:42:03 +02003234 eval 1 + 2
Bram Moolenaar2dd0a2c2020-08-08 15:10:27 +02003235 enddef
3236 func Func()
3237 noautocmd call s:AutocmdsDisabled()
3238 let g:ei_after = &eventignore
3239 endfunc
3240 Func()
3241 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003242 v9.CheckScriptSuccess(lines)
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02003243 g:ei_after->assert_equal('')
Bram Moolenaar2dd0a2c2020-08-08 15:10:27 +02003244enddef
3245
Bram Moolenaardfa3d552020-09-10 22:05:08 +02003246def StackTop()
Bram Moolenaarc3235272021-07-10 19:42:03 +02003247 eval 1 + 2
3248 eval 2 + 3
Bram Moolenaardfa3d552020-09-10 22:05:08 +02003249 # call not on fourth line
Bram Moolenaar62aec932022-01-29 21:45:34 +00003250 g:StackBot()
Bram Moolenaardfa3d552020-09-10 22:05:08 +02003251enddef
3252
3253def StackBot()
3254 # throw an error
3255 eval [][0]
3256enddef
3257
3258def Test_callstack_def()
3259 try
Bram Moolenaar62aec932022-01-29 21:45:34 +00003260 g:StackTop()
Bram Moolenaardfa3d552020-09-10 22:05:08 +02003261 catch
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02003262 v:throwpoint->assert_match('Test_callstack_def\[2\]..StackTop\[4\]..StackBot, line 2')
Bram Moolenaardfa3d552020-09-10 22:05:08 +02003263 endtry
3264enddef
3265
Bram Moolenaare8211a32020-10-09 22:04:29 +02003266" Re-using spot for variable used in block
3267def Test_block_scoped_var()
3268 var lines =<< trim END
3269 vim9script
3270 def Func()
3271 var x = ['a', 'b', 'c']
3272 if 1
3273 var y = 'x'
Bram Moolenaarbb8a7ce2021-04-10 20:10:26 +02003274 map(x, (_, _) => y)
Bram Moolenaare8211a32020-10-09 22:04:29 +02003275 endif
3276 var z = x
3277 assert_equal(['x', 'x', 'x'], z)
3278 enddef
3279 Func()
3280 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003281 v9.CheckScriptSuccess(lines)
Bram Moolenaare8211a32020-10-09 22:04:29 +02003282enddef
3283
Bram Moolenaareeece9e2020-11-20 19:26:48 +01003284def Test_reset_did_emsg()
3285 var lines =<< trim END
3286 @s = 'blah'
3287 au BufWinLeave * #
3288 def Func()
3289 var winid = popup_create('popup', {})
3290 exe '*s'
3291 popup_close(winid)
3292 enddef
3293 Func()
3294 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003295 v9.CheckScriptFailure(lines, 'E492:', 8)
Bram Moolenaar2d870f82020-12-05 13:41:01 +01003296 delfunc! g:Func
Bram Moolenaareeece9e2020-11-20 19:26:48 +01003297enddef
3298
Bram Moolenaar57f799e2020-12-12 20:42:19 +01003299def Test_did_emsg_reset()
3300 # executing an autocommand resets did_emsg, this should not result in a
3301 # builtin function considered failing
3302 var lines =<< trim END
3303 vim9script
3304 au BufWinLeave * #
3305 def Func()
Bram Moolenaar767034c2021-04-09 17:24:52 +02003306 popup_menu('', {callback: (a, b) => popup_create('', {})->popup_close()})
Bram Moolenaar57f799e2020-12-12 20:42:19 +01003307 eval [][0]
3308 enddef
3309 nno <F3> <cmd>call <sid>Func()<cr>
3310 feedkeys("\<F3>\e", 'xt')
3311 END
3312 writefile(lines, 'XemsgReset')
3313 assert_fails('so XemsgReset', ['E684:', 'E684:'], lines, 2)
3314 delete('XemsgReset')
3315 nunmap <F3>
3316 au! BufWinLeave
3317enddef
3318
Bram Moolenaar56602ba2020-12-05 21:22:08 +01003319def Test_abort_with_silent_call()
3320 var lines =<< trim END
3321 vim9script
3322 g:result = 'none'
3323 def Func()
3324 g:result += 3
3325 g:result = 'yes'
3326 enddef
3327 # error is silenced, but function aborts on error
3328 silent! Func()
3329 assert_equal('none', g:result)
3330 unlet g:result
3331 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003332 v9.CheckScriptSuccess(lines)
Bram Moolenaar56602ba2020-12-05 21:22:08 +01003333enddef
3334
Bram Moolenaarf665e972020-12-05 19:17:16 +01003335def Test_continues_with_silent_error()
3336 var lines =<< trim END
3337 vim9script
3338 g:result = 'none'
3339 def Func()
3340 silent! g:result += 3
3341 g:result = 'yes'
3342 enddef
3343 # error is silenced, function does not abort
3344 Func()
3345 assert_equal('yes', g:result)
3346 unlet g:result
3347 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003348 v9.CheckScriptSuccess(lines)
Bram Moolenaarf665e972020-12-05 19:17:16 +01003349enddef
3350
Bram Moolenaaraf0df472020-12-02 20:51:22 +01003351def Test_abort_even_with_silent()
3352 var lines =<< trim END
3353 vim9script
3354 g:result = 'none'
3355 def Func()
3356 eval {-> ''}() .. '' .. {}['X']
3357 g:result = 'yes'
3358 enddef
Bram Moolenaarf665e972020-12-05 19:17:16 +01003359 silent! Func()
Bram Moolenaaraf0df472020-12-02 20:51:22 +01003360 assert_equal('none', g:result)
Bram Moolenaar4029cab2020-12-05 18:13:27 +01003361 unlet g:result
3362 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003363 v9.CheckScriptSuccess(lines)
Bram Moolenaar4029cab2020-12-05 18:13:27 +01003364enddef
3365
Bram Moolenaarf665e972020-12-05 19:17:16 +01003366def Test_cmdmod_silent_restored()
3367 var lines =<< trim END
3368 vim9script
3369 def Func()
3370 g:result = 'none'
3371 silent! g:result += 3
3372 g:result = 'none'
3373 g:result += 3
3374 enddef
3375 Func()
3376 END
3377 # can't use CheckScriptFailure, it ignores the :silent!
3378 var fname = 'Xdefsilent'
3379 writefile(lines, fname)
3380 var caught = 'no'
3381 try
3382 exe 'source ' .. fname
3383 catch /E1030:/
3384 caught = 'yes'
3385 assert_match('Func, line 4', v:throwpoint)
3386 endtry
3387 assert_equal('yes', caught)
3388 delete(fname)
3389enddef
3390
Bram Moolenaar2fecb532021-03-24 22:00:56 +01003391def Test_cmdmod_silent_nested()
3392 var lines =<< trim END
3393 vim9script
3394 var result = ''
3395
3396 def Error()
3397 result ..= 'Eb'
3398 eval [][0]
3399 result ..= 'Ea'
3400 enddef
3401
3402 def Crash()
3403 result ..= 'Cb'
3404 sil! Error()
3405 result ..= 'Ca'
3406 enddef
3407
3408 Crash()
3409 assert_equal('CbEbEaCa', result)
3410 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003411 v9.CheckScriptSuccess(lines)
Bram Moolenaar2fecb532021-03-24 22:00:56 +01003412enddef
3413
Bram Moolenaar4029cab2020-12-05 18:13:27 +01003414def Test_dict_member_with_silent()
3415 var lines =<< trim END
3416 vim9script
3417 g:result = 'none'
3418 var d: dict<any>
3419 def Func()
3420 try
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01003421 g:result = map([], (_, v) => ({}[v]))->join() .. d['']
Bram Moolenaar4029cab2020-12-05 18:13:27 +01003422 catch
3423 endtry
3424 enddef
3425 silent! Func()
3426 assert_equal('0', g:result)
3427 unlet g:result
Bram Moolenaaraf0df472020-12-02 20:51:22 +01003428 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003429 v9.CheckScriptSuccess(lines)
Bram Moolenaaraf0df472020-12-02 20:51:22 +01003430enddef
3431
Bram Moolenaarf9041332021-01-21 19:41:16 +01003432def Test_skip_cmds_with_silent()
3433 var lines =<< trim END
3434 vim9script
3435
3436 def Func(b: bool)
3437 Crash()
3438 enddef
3439
3440 def Crash()
3441 sil! :/not found/d _
3442 sil! :/not found/put _
3443 enddef
3444
3445 Func(true)
3446 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003447 v9.CheckScriptSuccess(lines)
Bram Moolenaarf9041332021-01-21 19:41:16 +01003448enddef
3449
Bram Moolenaar5b3d1bb2020-12-22 12:20:08 +01003450def Test_opfunc()
Bram Moolenaar848fadd2022-01-30 15:28:30 +00003451 nnoremap <F3> <cmd>set opfunc=g:Opfunc<cr>g@
Bram Moolenaar5b3d1bb2020-12-22 12:20:08 +01003452 def g:Opfunc(_: any): string
3453 setline(1, 'ASDF')
3454 return ''
3455 enddef
3456 new
3457 setline(1, 'asdf')
3458 feedkeys("\<F3>$", 'x')
3459 assert_equal('ASDF', getline(1))
3460
3461 bwipe!
3462 nunmap <F3>
3463enddef
3464
Bram Moolenaar3b309f12021-12-13 18:19:55 +00003465func Test_opfunc_error()
3466 CheckScreendump
3467 call Run_Test_opfunc_error()
3468endfunc
3469
3470def Run_Test_opfunc_error()
3471 # test that the error from Opfunc() is displayed right away
3472 var lines =<< trim END
3473 vim9script
3474
3475 def Opfunc(type: string)
3476 try
3477 eval [][0]
3478 catch /nothing/ # error not caught
3479 endtry
3480 enddef
3481 &operatorfunc = Opfunc
3482 nnoremap <expr> l <SID>L()
3483 def L(): string
3484 return 'l'
3485 enddef
3486 'x'->repeat(10)->setline(1)
3487 feedkeys('g@l', 'n')
3488 feedkeys('llll')
3489 END
3490 call writefile(lines, 'XTest_opfunc_error')
3491
Bram Moolenaar62aec932022-01-29 21:45:34 +00003492 var buf = g:RunVimInTerminal('-S XTest_opfunc_error', {rows: 6, wait_for_ruler: 0})
3493 g:WaitForAssert(() => assert_match('Press ENTER', term_getline(buf, 6)))
3494 g:WaitForAssert(() => assert_match('E684: list index out of range: 0', term_getline(buf, 5)))
Bram Moolenaar3b309f12021-12-13 18:19:55 +00003495
3496 # clean up
Bram Moolenaar62aec932022-01-29 21:45:34 +00003497 g:StopVimInTerminal(buf)
Bram Moolenaar3b309f12021-12-13 18:19:55 +00003498 delete('XTest_opfunc_error')
3499enddef
3500
Bram Moolenaar077a4232020-12-22 18:33:27 +01003501" this was crashing on exit
3502def Test_nested_lambda_in_closure()
3503 var lines =<< trim END
3504 vim9script
Bram Moolenaar227c58a2021-04-28 20:40:44 +02003505 command WriteDone writefile(['Done'], 'XnestedDone')
Bram Moolenaar077a4232020-12-22 18:33:27 +01003506 def Outer()
3507 def g:Inner()
3508 echo map([1, 2, 3], {_, v -> v + 1})
3509 enddef
3510 g:Inner()
3511 enddef
3512 defcompile
Bram Moolenaar227c58a2021-04-28 20:40:44 +02003513 # not reached
Bram Moolenaar077a4232020-12-22 18:33:27 +01003514 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003515 if !g:RunVim([], lines, '--clean -c WriteDone -c quit')
Bram Moolenaar077a4232020-12-22 18:33:27 +01003516 return
3517 endif
3518 assert_equal(['Done'], readfile('XnestedDone'))
3519 delete('XnestedDone')
3520enddef
3521
Bram Moolenaar92368aa2022-02-07 17:50:39 +00003522def Test_nested_closure_funcref()
3523 var lines =<< trim END
3524 vim9script
3525 def Func()
3526 var n: number
3527 def Nested()
3528 ++n
3529 enddef
3530 Nested()
3531 g:result_one = n
3532 var Ref = function(Nested)
3533 Ref()
3534 g:result_two = n
3535 enddef
3536 Func()
3537 END
3538 v9.CheckScriptSuccess(lines)
3539 assert_equal(1, g:result_one)
3540 assert_equal(2, g:result_two)
3541 unlet g:result_one g:result_two
3542enddef
3543
Bram Moolenaar7aca5ca2022-02-07 19:56:43 +00003544def Test_nested_closure_in_dict()
3545 var lines =<< trim END
3546 vim9script
3547 def Func(): dict<any>
3548 var n: number
3549 def Inc(): number
3550 ++n
3551 return n
3552 enddef
3553 return {inc: function(Inc)}
3554 enddef
3555 disas Func
3556 var d = Func()
3557 assert_equal(1, d.inc())
3558 assert_equal(2, d.inc())
3559 END
3560 v9.CheckScriptSuccess(lines)
3561enddef
3562
Bram Moolenaar04947cc2021-03-06 19:26:46 +01003563def Test_check_func_arg_types()
3564 var lines =<< trim END
3565 vim9script
3566 def F1(x: string): string
3567 return x
3568 enddef
3569
3570 def F2(x: number): number
3571 return x + 1
3572 enddef
3573
3574 def G(g: func): dict<func>
3575 return {f: g}
3576 enddef
3577
3578 def H(d: dict<func>): string
3579 return d.f('a')
3580 enddef
3581 END
3582
Bram Moolenaar62aec932022-01-29 21:45:34 +00003583 v9.CheckScriptSuccess(lines + ['echo H(G(F1))'])
3584 v9.CheckScriptFailure(lines + ['echo H(G(F2))'], 'E1013:')
Bram Moolenaar04947cc2021-03-06 19:26:46 +01003585enddef
3586
Bram Moolenaar6e48b842021-08-10 22:52:02 +02003587def Test_list_any_type_checked()
3588 var lines =<< trim END
3589 vim9script
3590 def Foo()
3591 --decl--
3592 Bar(l)
3593 enddef
3594 def Bar(ll: list<dict<any>>)
3595 enddef
3596 Foo()
3597 END
Bram Moolenaar2d3ac2e2022-02-03 12:34:05 +00003598 # "any" could be "dict<any>", thus OK
Bram Moolenaar6e48b842021-08-10 22:52:02 +02003599 lines[2] = 'var l: list<any>'
Bram Moolenaar2d3ac2e2022-02-03 12:34:05 +00003600 v9.CheckScriptSuccess(lines)
Bram Moolenaar6e48b842021-08-10 22:52:02 +02003601 lines[2] = 'var l: list<any> = []'
Bram Moolenaar2d3ac2e2022-02-03 12:34:05 +00003602 v9.CheckScriptSuccess(lines)
Bram Moolenaar6e48b842021-08-10 22:52:02 +02003603
3604 lines[2] = 'var l: list<any> = [11]'
Bram Moolenaar62aec932022-01-29 21:45:34 +00003605 v9.CheckScriptFailure(lines, 'E1013: Argument 1: type mismatch, expected list<dict<any>> but got list<number>', 2)
Bram Moolenaar6e48b842021-08-10 22:52:02 +02003606enddef
3607
Bram Moolenaar701cc6c2021-04-10 13:33:48 +02003608def Test_compile_error()
3609 var lines =<< trim END
3610 def g:Broken()
3611 echo 'a' + {}
3612 enddef
3613 call g:Broken()
3614 END
3615 # First call: compilation error
Bram Moolenaar62aec932022-01-29 21:45:34 +00003616 v9.CheckScriptFailure(lines, 'E1051: Wrong argument type for +')
Bram Moolenaar701cc6c2021-04-10 13:33:48 +02003617
3618 # Second call won't try compiling again
3619 assert_fails('call g:Broken()', 'E1091: Function is not compiled: Broken')
Bram Moolenaar599410c2021-04-10 14:03:43 +02003620 delfunc g:Broken
3621
3622 # No error when compiling with :silent!
3623 lines =<< trim END
3624 def g:Broken()
3625 echo 'a' + []
3626 enddef
3627 silent! defcompile
3628 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003629 v9.CheckScriptSuccess(lines)
Bram Moolenaar599410c2021-04-10 14:03:43 +02003630
3631 # Calling the function won't try compiling again
3632 assert_fails('call g:Broken()', 'E1091: Function is not compiled: Broken')
3633 delfunc g:Broken
Bram Moolenaar701cc6c2021-04-10 13:33:48 +02003634enddef
3635
Bram Moolenaar962c43b2021-04-10 17:18:09 +02003636def Test_ignored_argument()
3637 var lines =<< trim END
3638 vim9script
3639 def Ignore(_, _): string
3640 return 'yes'
3641 enddef
3642 assert_equal('yes', Ignore(1, 2))
3643
3644 func Ok(_)
3645 return a:_
3646 endfunc
3647 assert_equal('ok', Ok('ok'))
3648
3649 func Oktoo()
3650 let _ = 'too'
3651 return _
3652 endfunc
3653 assert_equal('too', Oktoo())
Bram Moolenaarda479c72021-04-10 21:01:38 +02003654
3655 assert_equal([[1], [2], [3]], range(3)->mapnew((_, v) => [v]->map((_, w) => w + 1)))
Bram Moolenaar962c43b2021-04-10 17:18:09 +02003656 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003657 v9.CheckScriptSuccess(lines)
Bram Moolenaar962c43b2021-04-10 17:18:09 +02003658
3659 lines =<< trim END
3660 def Ignore(_: string): string
3661 return _
3662 enddef
3663 defcompile
3664 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003665 v9.CheckScriptFailure(lines, 'E1181:', 1)
Bram Moolenaar962c43b2021-04-10 17:18:09 +02003666
3667 lines =<< trim END
3668 var _ = 1
3669 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003670 v9.CheckDefAndScriptFailure(lines, 'E1181:', 1)
Yegappan Lakshmanan34fcb692021-05-25 20:14:00 +02003671
3672 lines =<< trim END
3673 var x = _
3674 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003675 v9.CheckDefAndScriptFailure(lines, 'E1181:', 1)
Bram Moolenaar962c43b2021-04-10 17:18:09 +02003676enddef
3677
Bram Moolenaarbb8a7ce2021-04-10 20:10:26 +02003678def Test_too_many_arguments()
3679 var lines =<< trim END
3680 echo [0, 1, 2]->map(() => 123)
3681 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003682 v9.CheckDefExecAndScriptFailure(lines, 'E1106: 2 arguments too many', 1)
Bram Moolenaarbb8a7ce2021-04-10 20:10:26 +02003683
3684 lines =<< trim END
3685 echo [0, 1, 2]->map((_) => 123)
3686 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003687 v9.CheckDefExecAndScriptFailure(lines, 'E1106: One argument too many', 1)
Bram Moolenaarbb8a7ce2021-04-10 20:10:26 +02003688enddef
Bram Moolenaar077a4232020-12-22 18:33:27 +01003689
Bram Moolenaara6aa1642021-04-23 19:32:23 +02003690def Test_closing_brace_at_start_of_line()
3691 var lines =<< trim END
3692 def Func()
3693 enddef
3694 Func(
3695 )
3696 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003697 v9.CheckDefAndScriptSuccess(lines)
Bram Moolenaara6aa1642021-04-23 19:32:23 +02003698enddef
3699
Bram Moolenaar62aec932022-01-29 21:45:34 +00003700func s:CreateMydict()
Bram Moolenaarb033ee22021-08-15 16:08:36 +02003701 let g:mydict = {}
3702 func g:mydict.afunc()
3703 let g:result = self.key
3704 endfunc
3705endfunc
3706
3707def Test_numbered_function_reference()
3708 CreateMydict()
3709 var output = execute('legacy func g:mydict.afunc')
3710 var funcName = 'g:' .. substitute(output, '.*function \(\d\+\).*', '\1', '')
3711 execute 'function(' .. funcName .. ', [], {key: 42})()'
3712 # check that the function still exists
3713 assert_equal(output, execute('legacy func g:mydict.afunc'))
3714 unlet g:mydict
3715enddef
3716
Bram Moolenaard3a11782022-01-05 16:50:40 +00003717def Test_go_beyond_end_of_cmd()
3718 # this was reading the byte after the end of the line
3719 var lines =<< trim END
3720 def F()
3721 cal
3722 enddef
3723 defcompile
3724 END
Bram Moolenaar62aec932022-01-29 21:45:34 +00003725 v9.CheckScriptFailure(lines, 'E476:')
Bram Moolenaard3a11782022-01-05 16:50:40 +00003726enddef
3727
Bram Moolenaar20677332021-06-06 17:02:53 +02003728if has('python3')
3729 def Test_python3_heredoc()
3730 py3 << trim EOF
3731 import vim
3732 vim.vars['didit'] = 'yes'
3733 EOF
3734 assert_equal('yes', g:didit)
3735
3736 python3 << trim EOF
3737 import vim
3738 vim.vars['didit'] = 'again'
3739 EOF
3740 assert_equal('again', g:didit)
3741 enddef
3742endif
3743
3744" This messes up syntax highlight, keep near the end.
3745if has('lua')
3746 def Test_lua_heredoc()
3747 g:d = {}
3748 lua << trim EOF
3749 x = vim.eval('g:d')
3750 x['key'] = 'val'
3751 EOF
3752 assert_equal('val', g:d.key)
3753 enddef
3754endif
3755
Bram Moolenaarf7779c62020-05-03 15:38:16 +02003756
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02003757" vim: ts=8 sw=2 sts=2 expandtab tw=80 fdm=marker