blob: 9f0cd3f895f4ff0006f41ff2b34e397eb19b042f [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 Moolenaar04b12692020-05-04 23:24:44 +02006source vim9.vim
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 Moolenaarf4e8cdd2020-10-12 22:07:13 +020033 var buf = RunVimInTerminal('-S XTest_compile_error',
Bram Moolenaare0de1712020-12-02 17:36:54 +010034 {rows: 10, wait_for_ruler: 0})
Bram Moolenaare8c46602021-04-05 22:27:37 +020035 WaitForAssert(() => assert_match('Error detected while compiling command line.*Fails.*Variable not found: nothing',
Bram Moolenaar03dfde22021-02-14 13:17:22 +010036 Term_getlines(buf, range(1, 9))))
Bram Moolenaarf4e8cdd2020-10-12 22:07:13 +020037
38 # clean up
Bram Moolenaare8c46602021-04-05 22:27:37 +020039 StopVimInTerminal(buf)
40 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
49 def script#OnlyCompiled()
50 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
67 var buf = RunVimInTerminal('-S XTest_compile_error', {rows: 10, wait_for_ruler: 0})
68 WaitForAssert(() => assert_match('Error detected while compiling command line.*function script#OnlyCompiled.*Invalid command: invalid',
69 Term_getlines(buf, range(1, 9))))
70
71 # clean up
72 StopVimInTerminal(buf)
73 delete('XTest_compile_error')
74 delete('Xdir', 'rf')
Bram Moolenaarf4e8cdd2020-10-12 22:07:13 +020075enddef
76
Bram Moolenaarf48b2fa2021-04-12 22:02:36 +020077def Test_autoload_name_mismatch()
78 var dir = 'Xdir/autoload'
79 mkdir(dir, 'p')
80
81 var lines =<< trim END
82 vim9script
83 def scriptX#Function()
84 # comment
85 g:runtime = 'yes'
86 enddef
87 END
88 writefile(lines, dir .. '/script.vim')
89
90 var save_rtp = &rtp
91 exe 'set rtp=' .. getcwd() .. '/Xdir'
92 lines =<< trim END
93 call script#Function()
94 END
95 CheckScriptFailure(lines, 'E746:', 2)
96
97 &rtp = save_rtp
98 delete(dir, 'rf')
99enddef
100
Bram Moolenaar0ba48e82020-11-17 18:23:19 +0100101def CallRecursive(n: number): number
102 return CallRecursive(n + 1)
103enddef
104
105def CallMapRecursive(l: list<number>): number
Bram Moolenaar2949cfd2020-12-31 21:28:47 +0100106 return map(l, (_, v) => CallMapRecursive([v]))[0]
Bram Moolenaar0ba48e82020-11-17 18:23:19 +0100107enddef
108
109def Test_funcdepth_error()
110 set maxfuncdepth=10
111
112 var caught = false
113 try
114 CallRecursive(1)
115 catch /E132:/
116 caught = true
117 endtry
118 assert_true(caught)
119
120 caught = false
121 try
122 CallMapRecursive([1])
123 catch /E132:/
124 caught = true
125 endtry
126 assert_true(caught)
127
128 set maxfuncdepth&
129enddef
130
Bram Moolenaar5178b1b2021-01-01 18:43:51 +0100131def Test_endfunc_enddef()
132 var lines =<< trim END
133 def Test()
134 echo 'test'
135 endfunc
136 enddef
137 END
138 CheckScriptFailure(lines, 'E1151:', 3)
139
140 lines =<< trim END
141 def Test()
142 func Nested()
143 echo 'test'
144 enddef
145 enddef
146 END
147 CheckScriptFailure(lines, 'E1152:', 4)
Bram Moolenaar49f1e9e2021-03-22 20:49:02 +0100148
149 lines =<< trim END
150 def Ok()
151 echo 'hello'
152 enddef | echo 'there'
153 def Bad()
154 echo 'hello'
155 enddef there
156 END
157 CheckScriptFailure(lines, 'E1173: Text found after enddef: there', 6)
Bram Moolenaar5178b1b2021-01-01 18:43:51 +0100158enddef
159
Bram Moolenaarb8ba9b92021-01-01 18:54:34 +0100160def Test_missing_endfunc_enddef()
161 var lines =<< trim END
162 vim9script
163 def Test()
164 echo 'test'
165 endef
166 END
167 CheckScriptFailure(lines, 'E1057:', 2)
168
169 lines =<< trim END
170 vim9script
171 func Some()
172 echo 'test'
173 enfffunc
174 END
175 CheckScriptFailure(lines, 'E126:', 2)
176enddef
177
Bram Moolenaar4efd9942021-01-24 21:14:20 +0100178def Test_white_space_before_paren()
179 var lines =<< trim END
180 vim9script
181 def Test ()
182 echo 'test'
183 enddef
184 END
185 CheckScriptFailure(lines, 'E1068:', 2)
186
187 lines =<< trim END
188 vim9script
189 func Test ()
190 echo 'test'
191 endfunc
192 END
193 CheckScriptFailure(lines, 'E1068:', 2)
194
195 lines =<< trim END
196 def Test ()
197 echo 'test'
198 enddef
199 END
200 CheckScriptFailure(lines, 'E1068:', 1)
201
202 lines =<< trim END
203 func Test ()
204 echo 'test'
205 endfunc
206 END
207 CheckScriptSuccess(lines)
208enddef
209
Bram Moolenaar832ea892021-01-08 21:55:26 +0100210def Test_enddef_dict_key()
211 var d = {
212 enddef: 'x',
213 endfunc: 'y',
214 }
215 assert_equal({enddef: 'x', endfunc: 'y'}, d)
216enddef
217
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200218def ReturnString(): string
219 return 'string'
220enddef
221
222def ReturnNumber(): number
223 return 123
224enddef
225
226let g:notNumber = 'string'
227
228def ReturnGlobal(): number
229 return g:notNumber
230enddef
231
232def Test_return_something()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +0200233 ReturnString()->assert_equal('string')
234 ReturnNumber()->assert_equal(123)
Bram Moolenaar5e654232020-09-16 15:22:00 +0200235 assert_fails('ReturnGlobal()', 'E1012: Type mismatch; expected number but got string', '', 1, 'ReturnGlobal')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200236enddef
237
Bram Moolenaare32e5162021-01-21 20:21:29 +0100238def Test_check_argument_type()
239 var lines =<< trim END
240 vim9script
241 def Val(a: number, b: number): number
242 return 0
243 enddef
244 def Func()
245 var x: any = true
246 Val(0, x)
247 enddef
248 disass Func
249 Func()
250 END
251 CheckScriptFailure(lines, 'E1013: Argument 2: type mismatch, expected number but got bool', 2)
252enddef
253
Bram Moolenaarefd88552020-06-18 20:50:10 +0200254def Test_missing_return()
255 CheckDefFailure(['def Missing(): number',
256 ' if g:cond',
257 ' echo "no return"',
258 ' else',
259 ' return 0',
260 ' endif'
261 'enddef'], 'E1027:')
262 CheckDefFailure(['def Missing(): number',
263 ' if g:cond',
264 ' return 1',
265 ' else',
266 ' echo "no return"',
267 ' endif'
268 'enddef'], 'E1027:')
269 CheckDefFailure(['def Missing(): number',
270 ' if g:cond',
271 ' return 1',
272 ' else',
273 ' return 2',
274 ' endif'
275 ' return 3'
276 'enddef'], 'E1095:')
277enddef
278
Bram Moolenaar403dc312020-10-17 19:29:51 +0200279def Test_return_bool()
280 var lines =<< trim END
281 vim9script
282 def MenuFilter(id: number, key: string): bool
283 return popup_filter_menu(id, key)
284 enddef
285 def YesnoFilter(id: number, key: string): bool
286 return popup_filter_yesno(id, key)
287 enddef
288 defcompile
289 END
290 CheckScriptSuccess(lines)
291enddef
292
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200293let s:nothing = 0
294def ReturnNothing()
295 s:nothing = 1
296 if true
297 return
298 endif
299 s:nothing = 2
300enddef
301
302def Test_return_nothing()
303 ReturnNothing()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +0200304 s:nothing->assert_equal(1)
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200305enddef
306
Bram Moolenaar648ea762021-01-15 19:04:32 +0100307def Test_return_invalid()
308 var lines =<< trim END
309 vim9script
310 def Func(): invalid
311 return xxx
312 enddef
313 defcompile
314 END
315 CheckScriptFailure(lines, 'E1010:', 2)
Bram Moolenaar31842cd2021-02-12 22:10:21 +0100316
317 lines =<< trim END
318 vim9script
319 def Test(Fun: func(number): number): list<number>
320 return map([1, 2, 3], (_, i) => Fun(i))
321 enddef
322 defcompile
323 def Inc(nr: number): nr
324 return nr + 2
325 enddef
326 echo Test(Inc)
327 END
328 # doing this twice was leaking memory
329 CheckScriptFailure(lines, 'E1010:')
330 CheckScriptFailure(lines, 'E1010:')
Bram Moolenaar648ea762021-01-15 19:04:32 +0100331enddef
332
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200333func Increment()
334 let g:counter += 1
335endfunc
336
337def Test_call_ufunc_count()
338 g:counter = 1
339 Increment()
340 Increment()
341 Increment()
Bram Moolenaarf5be8cd2020-07-17 20:36:00 +0200342 # works with and without :call
Bram Moolenaarc0c71e92020-09-11 19:09:48 +0200343 g:counter->assert_equal(4)
344 eval g:counter->assert_equal(4)
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200345 unlet g:counter
346enddef
347
348def MyVarargs(arg: string, ...rest: list<string>): string
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +0200349 var res = arg
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200350 for s in rest
351 res ..= ',' .. s
352 endfor
353 return res
354enddef
355
356def Test_call_varargs()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +0200357 MyVarargs('one')->assert_equal('one')
358 MyVarargs('one', 'two')->assert_equal('one,two')
359 MyVarargs('one', 'two', 'three')->assert_equal('one,two,three')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200360enddef
361
362def MyDefaultArgs(name = 'string'): string
363 return name
364enddef
365
Bram Moolenaare30f64b2020-07-15 19:48:20 +0200366def MyDefaultSecond(name: string, second: bool = true): string
367 return second ? name : 'none'
368enddef
369
Bram Moolenaar38a3bfa2021-03-29 22:14:55 +0200370
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200371def Test_call_default_args()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +0200372 MyDefaultArgs()->assert_equal('string')
Bram Moolenaar38a3bfa2021-03-29 22:14:55 +0200373 MyDefaultArgs(v:none)->assert_equal('string')
Bram Moolenaarc0c71e92020-09-11 19:09:48 +0200374 MyDefaultArgs('one')->assert_equal('one')
Bram Moolenaar38a3bfa2021-03-29 22:14:55 +0200375 assert_fails('MyDefaultArgs("one", "two")', 'E118:', '', 4, 'Test_call_default_args')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200376
Bram Moolenaarc0c71e92020-09-11 19:09:48 +0200377 MyDefaultSecond('test')->assert_equal('test')
378 MyDefaultSecond('test', true)->assert_equal('test')
379 MyDefaultSecond('test', false)->assert_equal('none')
Bram Moolenaare30f64b2020-07-15 19:48:20 +0200380
Bram Moolenaar38a3bfa2021-03-29 22:14:55 +0200381 var lines =<< trim END
382 def MyDefaultThird(name: string, aa = 'aa', bb = 'bb'): string
383 return name .. aa .. bb
384 enddef
385
386 MyDefaultThird('->')->assert_equal('->aabb')
387 MyDefaultThird('->', v:none)->assert_equal('->aabb')
388 MyDefaultThird('->', 'xx')->assert_equal('->xxbb')
389 MyDefaultThird('->', v:none, v:none)->assert_equal('->aabb')
390 MyDefaultThird('->', 'xx', v:none)->assert_equal('->xxbb')
391 MyDefaultThird('->', v:none, 'yy')->assert_equal('->aayy')
392 MyDefaultThird('->', 'xx', 'yy')->assert_equal('->xxyy')
393 END
394 CheckDefAndScriptSuccess(lines)
395
Bram Moolenaar822ba242020-05-24 23:00:18 +0200396 CheckScriptFailure(['def Func(arg: number = asdf)', 'enddef', 'defcompile'], 'E1001:')
Bram Moolenaar2d870f82020-12-05 13:41:01 +0100397 delfunc g:Func
Bram Moolenaar77072282020-09-16 17:55:40 +0200398 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 +0100399 delfunc g:Func
Yegappan Lakshmanan34fcb692021-05-25 20:14:00 +0200400 CheckDefFailure(['def Func(x: number = )', 'enddef'], 'E15:')
Bram Moolenaar12bce952021-03-11 20:04:04 +0100401
Bram Moolenaar38a3bfa2021-03-29 22:14:55 +0200402 lines =<< trim END
Bram Moolenaar12bce952021-03-11 20:04:04 +0100403 vim9script
404 def Func(a = b == 0 ? 1 : 2, b = 0)
405 enddef
406 defcompile
407 END
408 CheckScriptFailure(lines, 'E1001: Variable not found: b')
Bram Moolenaar04b12692020-05-04 23:24:44 +0200409enddef
410
Bram Moolenaarcef12702021-01-04 14:09:43 +0100411def FuncWithComment( # comment
412 a: number, #comment
413 b: bool, # comment
414 c: string) #comment
415 assert_equal(4, a)
416 assert_equal(true, b)
417 assert_equal('yes', c)
418enddef
419
420def Test_func_with_comments()
421 FuncWithComment(4, true, 'yes')
422
423 var lines =<< trim END
424 def Func(# comment
425 arg: string)
426 enddef
427 END
428 CheckScriptFailure(lines, 'E125:', 1)
429
430 lines =<< trim END
431 def Func(
432 arg: string# comment
433 )
434 enddef
435 END
436 CheckScriptFailure(lines, 'E475:', 2)
437
438 lines =<< trim END
439 def Func(
440 arg: string
441 )# comment
442 enddef
443 END
444 CheckScriptFailure(lines, 'E488:', 3)
445enddef
446
Bram Moolenaar04b12692020-05-04 23:24:44 +0200447def Test_nested_function()
448 def Nested(arg: string): string
449 return 'nested ' .. arg
450 enddef
Bram Moolenaarc0c71e92020-09-11 19:09:48 +0200451 Nested('function')->assert_equal('nested function')
Bram Moolenaar04b12692020-05-04 23:24:44 +0200452
Bram Moolenaar0e65d3d2020-05-05 17:53:16 +0200453 CheckDefFailure(['def Nested()', 'enddef', 'Nested(66)'], 'E118:')
454 CheckDefFailure(['def Nested(arg: string)', 'enddef', 'Nested()'], 'E119:')
455
Bram Moolenaar04b12692020-05-04 23:24:44 +0200456 CheckDefFailure(['func Nested()', 'endfunc'], 'E1086:')
Bram Moolenaarbcbf4132020-08-01 22:35:13 +0200457 CheckDefFailure(['def s:Nested()', 'enddef'], 'E1075:')
458 CheckDefFailure(['def b:Nested()', 'enddef'], 'E1075:')
Bram Moolenaar8b848ca2020-09-10 22:28:01 +0200459
Bram Moolenaar54021752020-12-06 18:50:36 +0100460 var lines =<< trim END
461 def Outer()
462 def Inner()
463 # comment
464 enddef
465 def Inner()
466 enddef
467 enddef
468 END
469 CheckDefFailure(lines, 'E1073:')
470
471 lines =<< trim END
472 def Outer()
473 def Inner()
474 # comment
475 enddef
476 def! Inner()
477 enddef
478 enddef
479 END
480 CheckDefFailure(lines, 'E1117:')
481
482 # nested function inside conditional
Bram Moolenaar54021752020-12-06 18:50:36 +0100483 lines =<< trim END
484 vim9script
485 var thecount = 0
486 if true
487 def Test(): number
488 def TheFunc(): number
489 thecount += 1
490 return thecount
491 enddef
492 return TheFunc()
493 enddef
494 endif
495 defcompile
496 assert_equal(1, Test())
497 assert_equal(2, Test())
498 END
499 CheckScriptSuccess(lines)
Bram Moolenaar8863bda2021-03-17 18:42:08 +0100500
501 # also works when "thecount" is inside the "if" block
502 lines =<< trim END
503 vim9script
504 if true
505 var thecount = 0
506 def Test(): number
507 def TheFunc(): number
508 thecount += 1
509 return thecount
510 enddef
511 return TheFunc()
512 enddef
513 endif
514 defcompile
515 assert_equal(1, Test())
516 assert_equal(2, Test())
517 END
518 CheckScriptSuccess(lines)
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200519enddef
520
Bram Moolenaaradc8e442020-12-31 18:28:18 +0100521def Test_not_nested_function()
522 echo printf('%d',
523 function('len')('xxx'))
524enddef
525
Bram Moolenaaraf8edbb2020-08-01 00:03:09 +0200526func Test_call_default_args_from_func()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +0200527 call MyDefaultArgs()->assert_equal('string')
528 call MyDefaultArgs('one')->assert_equal('one')
Bram Moolenaar9bd5d872020-09-06 21:47:48 +0200529 call assert_fails('call MyDefaultArgs("one", "two")', 'E118:', '', 3, 'Test_call_default_args_from_func')
Bram Moolenaaraf8edbb2020-08-01 00:03:09 +0200530endfunc
531
Bram Moolenaar38ddf332020-07-31 22:05:04 +0200532def Test_nested_global_function()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +0200533 var lines =<< trim END
Bram Moolenaar38ddf332020-07-31 22:05:04 +0200534 vim9script
535 def Outer()
536 def g:Inner(): string
537 return 'inner'
538 enddef
539 enddef
Bram Moolenaaraf8edbb2020-08-01 00:03:09 +0200540 defcompile
541 Outer()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +0200542 g:Inner()->assert_equal('inner')
Bram Moolenaaraf8edbb2020-08-01 00:03:09 +0200543 delfunc g:Inner
544 Outer()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +0200545 g:Inner()->assert_equal('inner')
Bram Moolenaaraf8edbb2020-08-01 00:03:09 +0200546 delfunc g:Inner
547 Outer()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +0200548 g:Inner()->assert_equal('inner')
Bram Moolenaaraf8edbb2020-08-01 00:03:09 +0200549 delfunc g:Inner
Bram Moolenaar38ddf332020-07-31 22:05:04 +0200550 END
551 CheckScriptSuccess(lines)
Bram Moolenaar2c79e9d2020-08-01 18:57:52 +0200552
553 lines =<< trim END
554 vim9script
555 def Outer()
556 def g:Inner(): string
557 return 'inner'
558 enddef
559 enddef
560 defcompile
561 Outer()
562 Outer()
563 END
564 CheckScriptFailure(lines, "E122:")
Bram Moolenaarcd45ed02020-12-22 17:35:54 +0100565 delfunc g:Inner
Bram Moolenaarad486a02020-08-01 23:22:18 +0200566
567 lines =<< trim END
568 vim9script
Bram Moolenaar58a52f22020-12-22 18:56:55 +0100569 def Outer()
570 def g:Inner()
Bram Moolenaar2949cfd2020-12-31 21:28:47 +0100571 echo map([1, 2, 3], (_, v) => v + 1)
Bram Moolenaar58a52f22020-12-22 18:56:55 +0100572 enddef
573 g:Inner()
574 enddef
575 Outer()
576 END
577 CheckScriptSuccess(lines)
578 delfunc g:Inner
579
580 lines =<< trim END
581 vim9script
Bram Moolenaarad486a02020-08-01 23:22:18 +0200582 def Func()
583 echo 'script'
584 enddef
585 def Outer()
586 def Func()
587 echo 'inner'
588 enddef
589 enddef
590 defcompile
591 END
592 CheckScriptFailure(lines, "E1073:")
Bram Moolenaar38ddf332020-07-31 22:05:04 +0200593enddef
594
Bram Moolenaar6abdcf82020-11-22 18:15:44 +0100595def DefListAll()
596 def
597enddef
598
599def DefListOne()
600 def DefListOne
601enddef
602
603def DefListMatches()
604 def /DefList
605enddef
606
607def Test_nested_def_list()
608 var funcs = split(execute('call DefListAll()'), "\n")
609 assert_true(len(funcs) > 10)
610 assert_true(funcs->index('def DefListAll()') >= 0)
611
612 funcs = split(execute('call DefListOne()'), "\n")
613 assert_equal([' def DefListOne()', '1 def DefListOne', ' enddef'], funcs)
614
615 funcs = split(execute('call DefListMatches()'), "\n")
616 assert_true(len(funcs) >= 3)
617 assert_true(funcs->index('def DefListAll()') >= 0)
618 assert_true(funcs->index('def DefListOne()') >= 0)
619 assert_true(funcs->index('def DefListMatches()') >= 0)
Bram Moolenaar54021752020-12-06 18:50:36 +0100620
621 var lines =<< trim END
622 vim9script
623 def Func()
624 def +Func+
625 enddef
626 defcompile
627 END
628 CheckScriptFailure(lines, 'E476:', 1)
Bram Moolenaar6abdcf82020-11-22 18:15:44 +0100629enddef
630
Bram Moolenaar333894b2020-08-01 18:53:07 +0200631def Test_global_local_function()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +0200632 var lines =<< trim END
Bram Moolenaar333894b2020-08-01 18:53:07 +0200633 vim9script
634 def g:Func(): string
635 return 'global'
636 enddef
637 def Func(): string
638 return 'local'
639 enddef
Bram Moolenaarc0c71e92020-09-11 19:09:48 +0200640 g:Func()->assert_equal('global')
641 Func()->assert_equal('local')
Bram Moolenaar2d870f82020-12-05 13:41:01 +0100642 delfunc g:Func
Bram Moolenaar333894b2020-08-01 18:53:07 +0200643 END
644 CheckScriptSuccess(lines)
Bram Moolenaar035d6e92020-08-11 22:30:42 +0200645
646 lines =<< trim END
647 vim9script
648 def g:Funcy()
649 echo 'funcy'
650 enddef
651 s:Funcy()
652 END
653 CheckScriptFailure(lines, 'E117:')
Bram Moolenaar333894b2020-08-01 18:53:07 +0200654enddef
655
Bram Moolenaar0f769812020-09-12 18:32:34 +0200656def Test_local_function_shadows_global()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +0200657 var lines =<< trim END
Bram Moolenaar0f769812020-09-12 18:32:34 +0200658 vim9script
659 def g:Gfunc(): string
660 return 'global'
661 enddef
662 def AnotherFunc(): number
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +0200663 var Gfunc = function('len')
Bram Moolenaar0f769812020-09-12 18:32:34 +0200664 return Gfunc('testing')
665 enddef
666 g:Gfunc()->assert_equal('global')
667 AnotherFunc()->assert_equal(7)
668 delfunc g:Gfunc
669 END
670 CheckScriptSuccess(lines)
671
672 lines =<< trim END
673 vim9script
674 def g:Func(): string
675 return 'global'
676 enddef
677 def AnotherFunc()
678 g:Func = function('len')
679 enddef
680 AnotherFunc()
681 END
682 CheckScriptFailure(lines, 'E705:')
683 delfunc g:Func
Bram Moolenaar0865b152021-04-05 15:38:51 +0200684
685 # global function is found without g: prefix
686 lines =<< trim END
687 vim9script
688 def g:Func(): string
689 return 'global'
690 enddef
691 def AnotherFunc(): string
692 return Func()
693 enddef
694 assert_equal('global', AnotherFunc())
695 delfunc g:Func
696 END
697 CheckScriptSuccess(lines)
698
699 lines =<< trim END
700 vim9script
701 def g:Func(): string
702 return 'global'
703 enddef
704 assert_equal('global', Func())
705 delfunc g:Func
706 END
707 CheckScriptSuccess(lines)
Bram Moolenaar0f769812020-09-12 18:32:34 +0200708enddef
709
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200710func TakesOneArg(arg)
711 echo a:arg
712endfunc
713
714def Test_call_wrong_args()
Bram Moolenaard2c61702020-09-06 15:58:36 +0200715 CheckDefFailure(['TakesOneArg()'], 'E119:')
716 CheckDefFailure(['TakesOneArg(11, 22)'], 'E118:')
717 CheckDefFailure(['bufnr(xxx)'], 'E1001:')
718 CheckScriptFailure(['def Func(Ref: func(s: string))'], 'E475:')
Bram Moolenaaree8580e2020-08-28 17:19:07 +0200719
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +0200720 var lines =<< trim END
Bram Moolenaaree8580e2020-08-28 17:19:07 +0200721 vim9script
722 def Func(s: string)
723 echo s
724 enddef
725 Func([])
726 END
Bram Moolenaar77072282020-09-16 17:55:40 +0200727 CheckScriptFailure(lines, 'E1013: Argument 1: type mismatch, expected string but got list<unknown>', 5)
Bram Moolenaarb185a402020-09-18 22:42:00 +0200728
729 lines =<< trim END
730 vim9script
Bram Moolenaarb4893b82021-02-21 22:20:24 +0100731 var name = 'piet'
732 def FuncOne(name: string)
733 echo nr
734 enddef
735 END
Bram Moolenaar057e84a2021-02-28 16:55:11 +0100736 CheckScriptFailure(lines, 'E1168:')
Bram Moolenaarb4893b82021-02-21 22:20:24 +0100737
738 lines =<< trim END
739 vim9script
Bram Moolenaarb185a402020-09-18 22:42:00 +0200740 def FuncOne(nr: number)
741 echo nr
742 enddef
743 def FuncTwo()
744 FuncOne()
745 enddef
746 defcompile
747 END
748 writefile(lines, 'Xscript')
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +0200749 var didCatch = false
Bram Moolenaarb185a402020-09-18 22:42:00 +0200750 try
751 source Xscript
752 catch
753 assert_match('E119: Not enough arguments for function: <SNR>\d\+_FuncOne', v:exception)
754 assert_match('Xscript\[8\]..function <SNR>\d\+_FuncTwo, line 1', v:throwpoint)
755 didCatch = true
756 endtry
757 assert_true(didCatch)
758
759 lines =<< trim END
760 vim9script
761 def FuncOne(nr: number)
762 echo nr
763 enddef
764 def FuncTwo()
765 FuncOne(1, 2)
766 enddef
767 defcompile
768 END
769 writefile(lines, 'Xscript')
770 didCatch = false
771 try
772 source Xscript
773 catch
774 assert_match('E118: Too many arguments for function: <SNR>\d\+_FuncOne', v:exception)
775 assert_match('Xscript\[8\]..function <SNR>\d\+_FuncTwo, line 1', v:throwpoint)
776 didCatch = true
777 endtry
778 assert_true(didCatch)
779
780 delete('Xscript')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200781enddef
782
Bram Moolenaar50824712020-12-20 21:10:17 +0100783def Test_call_funcref_wrong_args()
784 var head =<< trim END
785 vim9script
786 def Func3(a1: string, a2: number, a3: list<number>)
787 echo a1 .. a2 .. a3[0]
788 enddef
789 def Testme()
790 var funcMap: dict<func> = {func: Func3}
791 END
792 var tail =<< trim END
793 enddef
794 Testme()
795 END
796 CheckScriptSuccess(head + ["funcMap['func']('str', 123, [1, 2, 3])"] + tail)
797
798 CheckScriptFailure(head + ["funcMap['func']('str', 123)"] + tail, 'E119:')
799 CheckScriptFailure(head + ["funcMap['func']('str', 123, [1], 4)"] + tail, 'E118:')
Bram Moolenaar32b3f822021-01-06 21:59:39 +0100800
801 var lines =<< trim END
802 vim9script
803 var Ref: func(number): any
804 Ref = (j) => !j
805 echo Ref(false)
806 END
807 CheckScriptFailure(lines, 'E1013: Argument 1: type mismatch, expected number but got bool', 4)
808
809 lines =<< trim END
810 vim9script
811 var Ref: func(number): any
812 Ref = (j) => !j
813 call Ref(false)
814 END
815 CheckScriptFailure(lines, 'E1013: Argument 1: type mismatch, expected number but got bool', 4)
Bram Moolenaar50824712020-12-20 21:10:17 +0100816enddef
817
Bram Moolenaarb4d16cb2020-11-05 18:45:46 +0100818def Test_call_lambda_args()
Bram Moolenaar2a389082021-04-09 20:24:31 +0200819 var lines =<< trim END
820 var Callback = (..._) => 'anything'
821 assert_equal('anything', Callback())
822 assert_equal('anything', Callback(1))
823 assert_equal('anything', Callback('a', 2))
Bram Moolenaar1088b692021-04-09 22:12:44 +0200824
825 assert_equal('xyz', ((a: string): string => a)('xyz'))
Bram Moolenaar2a389082021-04-09 20:24:31 +0200826 END
827 CheckDefAndScriptSuccess(lines)
828
Bram Moolenaar2949cfd2020-12-31 21:28:47 +0100829 CheckDefFailure(['echo ((i) => 0)()'],
830 'E119: Not enough arguments for function: ((i) => 0)()')
Bram Moolenaarb4d16cb2020-11-05 18:45:46 +0100831
Bram Moolenaar2a389082021-04-09 20:24:31 +0200832 lines =<< trim END
Bram Moolenaar2949cfd2020-12-31 21:28:47 +0100833 var Ref = (x: number, y: number) => x + y
Bram Moolenaarb4d16cb2020-11-05 18:45:46 +0100834 echo Ref(1, 'x')
835 END
836 CheckDefFailure(lines, 'E1013: Argument 2: type mismatch, expected number but got string')
Bram Moolenaare68b02a2021-01-03 13:09:51 +0100837
838 lines =<< trim END
839 var Ref: func(job, string, number)
840 Ref = (x, y) => 0
841 END
842 CheckDefAndScriptFailure(lines, 'E1012:')
843
844 lines =<< trim END
845 var Ref: func(job, string)
846 Ref = (x, y, z) => 0
847 END
848 CheckDefAndScriptFailure(lines, 'E1012:')
Bram Moolenaar057e84a2021-02-28 16:55:11 +0100849
850 lines =<< trim END
851 var one = 1
852 var l = [1, 2, 3]
853 echo map(l, (one) => one)
854 END
855 CheckDefFailure(lines, 'E1167:')
856 CheckScriptFailure(['vim9script'] + lines, 'E1168:')
857
858 lines =<< trim END
859 def ShadowLocal()
860 var one = 1
861 var l = [1, 2, 3]
862 echo map(l, (one) => one)
863 enddef
864 END
865 CheckDefFailure(lines, 'E1167:')
866
867 lines =<< trim END
868 def Shadowarg(one: number)
869 var l = [1, 2, 3]
870 echo map(l, (one) => one)
871 enddef
872 END
873 CheckDefFailure(lines, 'E1167:')
Bram Moolenaar767034c2021-04-09 17:24:52 +0200874
875 lines =<< trim END
876 echo ((a) => a)('aa', 'bb')
877 END
878 CheckDefAndScriptFailure(lines, 'E118:', 1)
Bram Moolenaarb4d16cb2020-11-05 18:45:46 +0100879enddef
880
Bram Moolenaar5f91e742021-03-17 21:29:29 +0100881def FilterWithCond(x: string, Cond: func(string): bool): bool
882 return Cond(x)
883enddef
884
Bram Moolenaar0346b792021-01-31 22:18:29 +0100885def Test_lambda_return_type()
886 var lines =<< trim END
887 var Ref = (): => 123
888 END
889 CheckDefAndScriptFailure(lines, 'E1157:', 1)
Bram Moolenaar5f91e742021-03-17 21:29:29 +0100890
Yegappan Lakshmanan611728f2021-05-24 15:15:47 +0200891 # no space before the return type
892 lines =<< trim END
893 var Ref = (x):number => x + 1
894 END
895 CheckDefAndScriptFailure(lines, 'E1069:', 1)
896
Bram Moolenaar5f91e742021-03-17 21:29:29 +0100897 # this works
898 for x in ['foo', 'boo']
899 echo FilterWithCond(x, (v) => v =~ '^b')
900 endfor
901
902 # this fails
903 lines =<< trim END
904 echo FilterWithCond('foo', (v) => v .. '^b')
905 END
906 CheckDefAndScriptFailure(lines, 'E1013: Argument 2: type mismatch, expected func(string): bool but got func(any): string', 1)
Bram Moolenaar0346b792021-01-31 22:18:29 +0100907enddef
908
Bram Moolenaar709664c2020-12-12 14:33:41 +0100909def Test_lambda_uses_assigned_var()
910 CheckDefSuccess([
911 'var x: any = "aaa"'
Bram Moolenaar2949cfd2020-12-31 21:28:47 +0100912 'x = filter(["bbb"], (_, v) => v =~ x)'])
Bram Moolenaar709664c2020-12-12 14:33:41 +0100913enddef
914
Bram Moolenaar18062fc2021-03-05 21:35:47 +0100915def Test_pass_legacy_lambda_to_def_func()
916 var lines =<< trim END
917 vim9script
918 func Foo()
919 eval s:Bar({x -> 0})
920 endfunc
921 def Bar(y: any)
922 enddef
923 Foo()
924 END
925 CheckScriptSuccess(lines)
926enddef
927
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200928" Default arg and varargs
929def MyDefVarargs(one: string, two = 'foo', ...rest: list<string>): string
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +0200930 var res = one .. ',' .. two
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200931 for s in rest
932 res ..= ',' .. s
933 endfor
934 return res
935enddef
936
937def Test_call_def_varargs()
Bram Moolenaar9bd5d872020-09-06 21:47:48 +0200938 assert_fails('MyDefVarargs()', 'E119:', '', 1, 'Test_call_def_varargs')
Bram Moolenaarc0c71e92020-09-11 19:09:48 +0200939 MyDefVarargs('one')->assert_equal('one,foo')
940 MyDefVarargs('one', 'two')->assert_equal('one,two')
941 MyDefVarargs('one', 'two', 'three')->assert_equal('one,two,three')
Bram Moolenaar24aa48b2020-07-25 16:33:02 +0200942 CheckDefFailure(['MyDefVarargs("one", 22)'],
Bram Moolenaar77072282020-09-16 17:55:40 +0200943 'E1013: Argument 2: type mismatch, expected string but got number')
Bram Moolenaar24aa48b2020-07-25 16:33:02 +0200944 CheckDefFailure(['MyDefVarargs("one", "two", 123)'],
Bram Moolenaar77072282020-09-16 17:55:40 +0200945 'E1013: Argument 3: type mismatch, expected string but got number')
Bram Moolenaar24aa48b2020-07-25 16:33:02 +0200946
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +0200947 var lines =<< trim END
Bram Moolenaar24aa48b2020-07-25 16:33:02 +0200948 vim9script
949 def Func(...l: list<string>)
950 echo l
951 enddef
952 Func('a', 'b', 'c')
953 END
954 CheckScriptSuccess(lines)
955
956 lines =<< trim END
957 vim9script
958 def Func(...l: list<string>)
959 echo l
960 enddef
961 Func()
962 END
963 CheckScriptSuccess(lines)
964
965 lines =<< trim END
966 vim9script
Bram Moolenaar2a389082021-04-09 20:24:31 +0200967 def Func(...l: list<any>)
Bram Moolenaar2f8cbc42020-09-16 17:22:59 +0200968 echo l
969 enddef
970 Func(0)
971 END
972 CheckScriptSuccess(lines)
973
974 lines =<< trim END
975 vim9script
Bram Moolenaar2a389082021-04-09 20:24:31 +0200976 def Func(...l: any)
977 echo l
978 enddef
979 Func(0)
980 END
981 CheckScriptFailure(lines, 'E1180:', 2)
982
983 lines =<< trim END
984 vim9script
Bram Moolenaar28022722020-09-21 22:02:49 +0200985 def Func(..._l: list<string>)
986 echo _l
987 enddef
988 Func('a', 'b', 'c')
989 END
990 CheckScriptSuccess(lines)
991
992 lines =<< trim END
993 vim9script
Bram Moolenaar24aa48b2020-07-25 16:33:02 +0200994 def Func(...l: list<string>)
995 echo l
996 enddef
997 Func(1, 2, 3)
998 END
Bram Moolenaar77072282020-09-16 17:55:40 +0200999 CheckScriptFailure(lines, 'E1013: Argument 1: type mismatch')
Bram Moolenaar24aa48b2020-07-25 16:33:02 +02001000
1001 lines =<< trim END
1002 vim9script
1003 def Func(...l: list<string>)
1004 echo l
1005 enddef
1006 Func('a', 9)
1007 END
Bram Moolenaar77072282020-09-16 17:55:40 +02001008 CheckScriptFailure(lines, 'E1013: Argument 2: type mismatch')
Bram Moolenaar24aa48b2020-07-25 16:33:02 +02001009
1010 lines =<< trim END
1011 vim9script
1012 def Func(...l: list<string>)
1013 echo l
1014 enddef
1015 Func(1, 'a')
1016 END
Bram Moolenaar77072282020-09-16 17:55:40 +02001017 CheckScriptFailure(lines, 'E1013: Argument 1: type mismatch')
Bram Moolenaar4f53b792021-02-07 15:59:49 +01001018
1019 lines =<< trim END
1020 vim9script
1021 def Func( # some comment
1022 ...l = []
1023 )
1024 echo l
1025 enddef
1026 END
1027 CheckScriptFailure(lines, 'E1160:')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001028enddef
1029
Bram Moolenaar1378fbc2020-04-11 20:50:33 +02001030let s:value = ''
1031
1032def FuncOneDefArg(opt = 'text')
1033 s:value = opt
1034enddef
1035
1036def FuncTwoDefArg(nr = 123, opt = 'text'): string
1037 return nr .. opt
1038enddef
1039
1040def FuncVarargs(...arg: list<string>): string
1041 return join(arg, ',')
1042enddef
1043
1044def Test_func_type_varargs()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001045 var RefDefArg: func(?string)
Bram Moolenaar1378fbc2020-04-11 20:50:33 +02001046 RefDefArg = FuncOneDefArg
1047 RefDefArg()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001048 s:value->assert_equal('text')
Bram Moolenaar1378fbc2020-04-11 20:50:33 +02001049 RefDefArg('some')
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001050 s:value->assert_equal('some')
Bram Moolenaar1378fbc2020-04-11 20:50:33 +02001051
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001052 var RefDef2Arg: func(?number, ?string): string
Bram Moolenaar1378fbc2020-04-11 20:50:33 +02001053 RefDef2Arg = FuncTwoDefArg
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001054 RefDef2Arg()->assert_equal('123text')
1055 RefDef2Arg(99)->assert_equal('99text')
1056 RefDef2Arg(77, 'some')->assert_equal('77some')
Bram Moolenaar1378fbc2020-04-11 20:50:33 +02001057
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001058 CheckDefFailure(['var RefWrong: func(string?)'], 'E1010:')
1059 CheckDefFailure(['var RefWrong: func(?string, string)'], 'E1007:')
Bram Moolenaar1378fbc2020-04-11 20:50:33 +02001060
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001061 var RefVarargs: func(...list<string>): string
Bram Moolenaar1378fbc2020-04-11 20:50:33 +02001062 RefVarargs = FuncVarargs
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001063 RefVarargs()->assert_equal('')
1064 RefVarargs('one')->assert_equal('one')
1065 RefVarargs('one', 'two')->assert_equal('one,two')
Bram Moolenaar1378fbc2020-04-11 20:50:33 +02001066
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001067 CheckDefFailure(['var RefWrong: func(...list<string>, string)'], 'E110:')
1068 CheckDefFailure(['var RefWrong: func(...list<string>, ?string)'], 'E110:')
Bram Moolenaar1378fbc2020-04-11 20:50:33 +02001069enddef
1070
Bram Moolenaar0b76b422020-04-07 22:05:08 +02001071" Only varargs
1072def MyVarargsOnly(...args: list<string>): string
1073 return join(args, ',')
1074enddef
1075
1076def Test_call_varargs_only()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001077 MyVarargsOnly()->assert_equal('')
1078 MyVarargsOnly('one')->assert_equal('one')
1079 MyVarargsOnly('one', 'two')->assert_equal('one,two')
Bram Moolenaar77072282020-09-16 17:55:40 +02001080 CheckDefFailure(['MyVarargsOnly(1)'], 'E1013: Argument 1: type mismatch, expected string but got number')
1081 CheckDefFailure(['MyVarargsOnly("one", 2)'], 'E1013: Argument 2: type mismatch, expected string but got number')
Bram Moolenaar0b76b422020-04-07 22:05:08 +02001082enddef
1083
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001084def Test_using_var_as_arg()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001085 writefile(['def Func(x: number)', 'var x = 234', 'enddef', 'defcompile'], 'Xdef')
Bram Moolenaar9bd5d872020-09-06 21:47:48 +02001086 assert_fails('so Xdef', 'E1006:', '', 1, 'Func')
Bram Moolenaard2c61702020-09-06 15:58:36 +02001087 delete('Xdef')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001088enddef
1089
Bram Moolenaarcb2bdb12020-05-10 22:53:56 +02001090def DictArg(arg: dict<string>)
1091 arg['key'] = 'value'
1092enddef
1093
1094def ListArg(arg: list<string>)
1095 arg[0] = 'value'
1096enddef
1097
1098def Test_assign_to_argument()
Bram Moolenaarf5be8cd2020-07-17 20:36:00 +02001099 # works for dict and list
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001100 var d: dict<string> = {}
Bram Moolenaarcb2bdb12020-05-10 22:53:56 +02001101 DictArg(d)
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001102 d['key']->assert_equal('value')
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001103 var l: list<string> = []
Bram Moolenaarcb2bdb12020-05-10 22:53:56 +02001104 ListArg(l)
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001105 l[0]->assert_equal('value')
Bram Moolenaarcb2bdb12020-05-10 22:53:56 +02001106
Bram Moolenaard2c61702020-09-06 15:58:36 +02001107 CheckScriptFailure(['def Func(arg: number)', 'arg = 3', 'enddef', 'defcompile'], 'E1090:')
Bram Moolenaar2d870f82020-12-05 13:41:01 +01001108 delfunc! g:Func
Bram Moolenaarcb2bdb12020-05-10 22:53:56 +02001109enddef
1110
Bram Moolenaarb816dae2020-09-20 22:04:00 +02001111" These argument names are reserved in legacy functions.
1112def WithReservedNames(firstline: string, lastline: string): string
1113 return firstline .. lastline
1114enddef
1115
1116def Test_argument_names()
1117 assert_equal('OK', WithReservedNames('O', 'K'))
1118enddef
1119
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001120def Test_call_func_defined_later()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001121 g:DefinedLater('one')->assert_equal('one')
Bram Moolenaar9bd5d872020-09-06 21:47:48 +02001122 assert_fails('NotDefined("one")', 'E117:', '', 2, 'Test_call_func_defined_later')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001123enddef
1124
Bram Moolenaar1df8b3f2020-04-23 18:13:23 +02001125func DefinedLater(arg)
1126 return a:arg
1127endfunc
1128
1129def Test_call_funcref()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001130 g:SomeFunc('abc')->assert_equal(3)
Bram Moolenaar9bd5d872020-09-06 21:47:48 +02001131 assert_fails('NotAFunc()', 'E117:', '', 2, 'Test_call_funcref') # comment after call
1132 assert_fails('g:NotAFunc()', 'E117:', '', 3, 'Test_call_funcref')
Bram Moolenaar2f1980f2020-07-22 19:30:06 +02001133
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001134 var lines =<< trim END
Bram Moolenaar2f1980f2020-07-22 19:30:06 +02001135 vim9script
1136 def RetNumber(): number
1137 return 123
1138 enddef
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001139 var Funcref: func: number = function('RetNumber')
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001140 Funcref()->assert_equal(123)
Bram Moolenaar2f1980f2020-07-22 19:30:06 +02001141 END
1142 CheckScriptSuccess(lines)
Bram Moolenaar0f60e802020-07-22 20:16:11 +02001143
1144 lines =<< trim END
1145 vim9script
1146 def RetNumber(): number
1147 return 123
1148 enddef
1149 def Bar(F: func: number): number
1150 return F()
1151 enddef
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001152 var Funcref = function('RetNumber')
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001153 Bar(Funcref)->assert_equal(123)
Bram Moolenaar0f60e802020-07-22 20:16:11 +02001154 END
1155 CheckScriptSuccess(lines)
Bram Moolenaarbfba8652020-07-23 20:09:10 +02001156
1157 lines =<< trim END
1158 vim9script
1159 def UseNumber(nr: number)
1160 echo nr
1161 enddef
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001162 var Funcref: func(number) = function('UseNumber')
Bram Moolenaarbfba8652020-07-23 20:09:10 +02001163 Funcref(123)
1164 END
1165 CheckScriptSuccess(lines)
Bram Moolenaarb8070e32020-07-23 20:56:04 +02001166
1167 lines =<< trim END
1168 vim9script
1169 def UseNumber(nr: number)
1170 echo nr
1171 enddef
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001172 var Funcref: func(string) = function('UseNumber')
Bram Moolenaarb8070e32020-07-23 20:56:04 +02001173 END
Bram Moolenaar5e654232020-09-16 15:22:00 +02001174 CheckScriptFailure(lines, 'E1012: Type mismatch; expected func(string) but got func(number)')
Bram Moolenaar4fc224c2020-07-26 17:56:25 +02001175
1176 lines =<< trim END
1177 vim9script
1178 def EchoNr(nr = 34)
1179 g:echo = nr
1180 enddef
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001181 var Funcref: func(?number) = function('EchoNr')
Bram Moolenaar4fc224c2020-07-26 17:56:25 +02001182 Funcref()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001183 g:echo->assert_equal(34)
Bram Moolenaar4fc224c2020-07-26 17:56:25 +02001184 Funcref(123)
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001185 g:echo->assert_equal(123)
Bram Moolenaar4fc224c2020-07-26 17:56:25 +02001186 END
1187 CheckScriptSuccess(lines)
Bram Moolenaarace61322020-07-26 18:16:58 +02001188
1189 lines =<< trim END
1190 vim9script
1191 def EchoList(...l: list<number>)
1192 g:echo = l
1193 enddef
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001194 var Funcref: func(...list<number>) = function('EchoList')
Bram Moolenaarace61322020-07-26 18:16:58 +02001195 Funcref()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001196 g:echo->assert_equal([])
Bram Moolenaarace61322020-07-26 18:16:58 +02001197 Funcref(1, 2, 3)
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001198 g:echo->assert_equal([1, 2, 3])
Bram Moolenaarace61322020-07-26 18:16:58 +02001199 END
1200 CheckScriptSuccess(lines)
Bram Moolenaar01865ad2020-07-26 18:33:09 +02001201
1202 lines =<< trim END
1203 vim9script
1204 def OptAndVar(nr: number, opt = 12, ...l: list<number>): number
1205 g:optarg = opt
1206 g:listarg = l
1207 return nr
1208 enddef
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001209 var Funcref: func(number, ?number, ...list<number>): number = function('OptAndVar')
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001210 Funcref(10)->assert_equal(10)
1211 g:optarg->assert_equal(12)
1212 g:listarg->assert_equal([])
Bram Moolenaar01865ad2020-07-26 18:33:09 +02001213
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001214 Funcref(11, 22)->assert_equal(11)
1215 g:optarg->assert_equal(22)
1216 g:listarg->assert_equal([])
Bram Moolenaar01865ad2020-07-26 18:33:09 +02001217
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001218 Funcref(17, 18, 1, 2, 3)->assert_equal(17)
1219 g:optarg->assert_equal(18)
1220 g:listarg->assert_equal([1, 2, 3])
Bram Moolenaar01865ad2020-07-26 18:33:09 +02001221 END
1222 CheckScriptSuccess(lines)
Bram Moolenaar1df8b3f2020-04-23 18:13:23 +02001223enddef
1224
1225let SomeFunc = function('len')
1226let NotAFunc = 'text'
1227
Bram Moolenaar99aaf0c2020-04-12 14:39:53 +02001228def CombineFuncrefTypes()
Bram Moolenaarf5be8cd2020-07-17 20:36:00 +02001229 # same arguments, different return type
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001230 var Ref1: func(bool): string
1231 var Ref2: func(bool): number
1232 var Ref3: func(bool): any
Bram Moolenaar99aaf0c2020-04-12 14:39:53 +02001233 Ref3 = g:cond ? Ref1 : Ref2
1234
Bram Moolenaarf5be8cd2020-07-17 20:36:00 +02001235 # different number of arguments
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001236 var Refa1: func(bool): number
1237 var Refa2: func(bool, number): number
1238 var Refa3: func: number
Bram Moolenaar99aaf0c2020-04-12 14:39:53 +02001239 Refa3 = g:cond ? Refa1 : Refa2
1240
Bram Moolenaarf5be8cd2020-07-17 20:36:00 +02001241 # different argument types
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001242 var Refb1: func(bool, string): number
1243 var Refb2: func(string, number): number
1244 var Refb3: func(any, any): number
Bram Moolenaar99aaf0c2020-04-12 14:39:53 +02001245 Refb3 = g:cond ? Refb1 : Refb2
1246enddef
1247
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001248def FuncWithForwardCall()
Bram Moolenaar1df8b3f2020-04-23 18:13:23 +02001249 return g:DefinedEvenLater("yes")
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001250enddef
1251
1252def DefinedEvenLater(arg: string): string
1253 return arg
1254enddef
1255
1256def Test_error_in_nested_function()
Bram Moolenaarf5be8cd2020-07-17 20:36:00 +02001257 # Error in called function requires unwinding the call stack.
Bram Moolenaar44d66522020-09-06 22:26:57 +02001258 assert_fails('FuncWithForwardCall()', 'E1096:', '', 1, 'FuncWithForwardCall')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001259enddef
1260
1261def Test_return_type_wrong()
Bram Moolenaar5a849da2020-08-08 16:47:30 +02001262 CheckScriptFailure([
1263 'def Func(): number',
1264 'return "a"',
1265 'enddef',
1266 'defcompile'], 'expected number but got string')
Bram Moolenaar2d870f82020-12-05 13:41:01 +01001267 delfunc! g:Func
Bram Moolenaar5a849da2020-08-08 16:47:30 +02001268 CheckScriptFailure([
1269 'def Func(): string',
1270 'return 1',
1271 'enddef',
1272 'defcompile'], 'expected string but got number')
Bram Moolenaar2d870f82020-12-05 13:41:01 +01001273 delfunc! g:Func
Bram Moolenaar5a849da2020-08-08 16:47:30 +02001274 CheckScriptFailure([
1275 'def Func(): void',
1276 'return "a"',
1277 'enddef',
1278 'defcompile'],
1279 'E1096: Returning a value in a function without a return type')
Bram Moolenaar2d870f82020-12-05 13:41:01 +01001280 delfunc! g:Func
Bram Moolenaar5a849da2020-08-08 16:47:30 +02001281 CheckScriptFailure([
1282 'def Func()',
1283 'return "a"',
1284 'enddef',
1285 'defcompile'],
1286 'E1096: Returning a value in a function without a return type')
Bram Moolenaar2d870f82020-12-05 13:41:01 +01001287 delfunc! g:Func
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001288
Bram Moolenaar5a849da2020-08-08 16:47:30 +02001289 CheckScriptFailure([
1290 'def Func(): number',
1291 'return',
1292 'enddef',
1293 'defcompile'], 'E1003:')
Bram Moolenaar2d870f82020-12-05 13:41:01 +01001294 delfunc! g:Func
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001295
1296 CheckScriptFailure(['def Func(): list', 'return []', 'enddef'], 'E1008:')
Bram Moolenaar2d870f82020-12-05 13:41:01 +01001297 delfunc! g:Func
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001298 CheckScriptFailure(['def Func(): dict', 'return {}', 'enddef'], 'E1008:')
Bram Moolenaar2d870f82020-12-05 13:41:01 +01001299 delfunc! g:Func
Bram Moolenaaree4e0c12020-04-06 21:35:05 +02001300 CheckScriptFailure(['def Func()', 'return 1'], 'E1057:')
Bram Moolenaar2d870f82020-12-05 13:41:01 +01001301 delfunc! g:Func
Bram Moolenaar5a849da2020-08-08 16:47:30 +02001302
1303 CheckScriptFailure([
1304 'vim9script',
1305 'def FuncB()',
1306 ' return 123',
1307 'enddef',
1308 'def FuncA()',
1309 ' FuncB()',
1310 'enddef',
1311 'defcompile'], 'E1096:')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001312enddef
1313
1314def Test_arg_type_wrong()
1315 CheckScriptFailure(['def Func3(items: list)', 'echo "a"', 'enddef'], 'E1008: Missing <type>')
Bram Moolenaaree4e0c12020-04-06 21:35:05 +02001316 CheckScriptFailure(['def Func4(...)', 'echo "a"', 'enddef'], 'E1055: Missing name after ...')
Bram Moolenaarf93c7fe2020-04-23 22:16:53 +02001317 CheckScriptFailure(['def Func5(items:string)', 'echo "a"'], 'E1069:')
Bram Moolenaar6e949782020-04-13 17:21:00 +02001318 CheckScriptFailure(['def Func5(items)', 'echo "a"'], 'E1077:')
Yegappan Lakshmanan34fcb692021-05-25 20:14:00 +02001319 CheckScriptFailure(['def Func6(...x:list<number>)', 'echo "a"', 'enddef'], 'E1069:')
1320 CheckScriptFailure(['def Func7(...x: int)', 'echo "a"', 'enddef'], 'E1010:')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001321enddef
1322
Bram Moolenaar86cdb8a2021-04-06 19:01:03 +02001323def Test_white_space_before_comma()
1324 var lines =<< trim END
1325 vim9script
1326 def Func(a: number , b: number)
1327 enddef
1328 END
1329 CheckScriptFailure(lines, 'E1068:')
Yegappan Lakshmanan611728f2021-05-24 15:15:47 +02001330 call assert_fails('vim9cmd echo stridx("a" .. "b" , "a")', 'E1068:')
Bram Moolenaar86cdb8a2021-04-06 19:01:03 +02001331enddef
1332
Bram Moolenaar608d78f2021-03-06 22:33:12 +01001333def Test_white_space_after_comma()
1334 var lines =<< trim END
1335 vim9script
1336 def Func(a: number,b: number)
1337 enddef
1338 END
1339 CheckScriptFailure(lines, 'E1069:')
1340
1341 # OK in legacy function
1342 lines =<< trim END
1343 vim9script
1344 func Func(a,b)
1345 endfunc
1346 END
1347 CheckScriptSuccess(lines)
1348enddef
1349
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001350def Test_vim9script_call()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001351 var lines =<< trim END
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001352 vim9script
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001353 var name = ''
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001354 def MyFunc(arg: string)
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001355 name = arg
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001356 enddef
1357 MyFunc('foobar')
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001358 name->assert_equal('foobar')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001359
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001360 var str = 'barfoo'
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001361 str->MyFunc()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001362 name->assert_equal('barfoo')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001363
Bram Moolenaar67979662020-06-20 22:50:47 +02001364 g:value = 'value'
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001365 g:value->MyFunc()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001366 name->assert_equal('value')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001367
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001368 var listvar = []
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001369 def ListFunc(arg: list<number>)
1370 listvar = arg
1371 enddef
1372 [1, 2, 3]->ListFunc()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001373 listvar->assert_equal([1, 2, 3])
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001374
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001375 var dictvar = {}
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001376 def DictFunc(arg: dict<number>)
1377 dictvar = arg
1378 enddef
Bram Moolenaare0de1712020-12-02 17:36:54 +01001379 {a: 1, b: 2}->DictFunc()
1380 dictvar->assert_equal({a: 1, b: 2})
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001381 def CompiledDict()
Bram Moolenaare0de1712020-12-02 17:36:54 +01001382 {a: 3, b: 4}->DictFunc()
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001383 enddef
1384 CompiledDict()
Bram Moolenaare0de1712020-12-02 17:36:54 +01001385 dictvar->assert_equal({a: 3, b: 4})
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001386
Bram Moolenaare0de1712020-12-02 17:36:54 +01001387 {a: 3, b: 4}->DictFunc()
1388 dictvar->assert_equal({a: 3, b: 4})
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001389
1390 ('text')->MyFunc()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001391 name->assert_equal('text')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001392 ("some")->MyFunc()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001393 name->assert_equal('some')
Bram Moolenaare6b53242020-07-01 17:28:33 +02001394
Bram Moolenaar13e12b82020-07-24 18:47:22 +02001395 # line starting with single quote is not a mark
Bram Moolenaar10409562020-07-29 20:00:38 +02001396 # line starting with double quote can be a method call
Bram Moolenaar3d48e252020-07-15 14:15:52 +02001397 'asdfasdf'->MyFunc()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001398 name->assert_equal('asdfasdf')
Bram Moolenaar10409562020-07-29 20:00:38 +02001399 "xyz"->MyFunc()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001400 name->assert_equal('xyz')
Bram Moolenaar3d48e252020-07-15 14:15:52 +02001401
1402 def UseString()
1403 'xyork'->MyFunc()
1404 enddef
1405 UseString()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001406 name->assert_equal('xyork')
Bram Moolenaar3d48e252020-07-15 14:15:52 +02001407
Bram Moolenaar10409562020-07-29 20:00:38 +02001408 def UseString2()
1409 "knife"->MyFunc()
1410 enddef
1411 UseString2()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001412 name->assert_equal('knife')
Bram Moolenaar10409562020-07-29 20:00:38 +02001413
Bram Moolenaar13e12b82020-07-24 18:47:22 +02001414 # prepending a colon makes it a mark
1415 new
1416 setline(1, ['aaa', 'bbb', 'ccc'])
1417 normal! 3Gmt1G
1418 :'t
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001419 getcurpos()[1]->assert_equal(3)
Bram Moolenaar13e12b82020-07-24 18:47:22 +02001420 bwipe!
1421
Bram Moolenaare6b53242020-07-01 17:28:33 +02001422 MyFunc(
1423 'continued'
1424 )
1425 assert_equal('continued',
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001426 name
Bram Moolenaare6b53242020-07-01 17:28:33 +02001427 )
1428
1429 call MyFunc(
1430 'more'
1431 ..
1432 'lines'
1433 )
1434 assert_equal(
1435 'morelines',
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001436 name)
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001437 END
1438 writefile(lines, 'Xcall.vim')
1439 source Xcall.vim
1440 delete('Xcall.vim')
1441enddef
1442
1443def Test_vim9script_call_fail_decl()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001444 var lines =<< trim END
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001445 vim9script
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001446 var name = ''
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001447 def MyFunc(arg: string)
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001448 var name = 123
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001449 enddef
Bram Moolenaar822ba242020-05-24 23:00:18 +02001450 defcompile
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001451 END
Bram Moolenaar6c4bfe42020-07-23 18:26:30 +02001452 CheckScriptFailure(lines, 'E1054:')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001453enddef
1454
Bram Moolenaar65b95452020-07-19 14:03:09 +02001455def Test_vim9script_call_fail_type()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001456 var lines =<< trim END
Bram Moolenaar65b95452020-07-19 14:03:09 +02001457 vim9script
1458 def MyFunc(arg: string)
1459 echo arg
1460 enddef
1461 MyFunc(1234)
1462 END
Bram Moolenaar77072282020-09-16 17:55:40 +02001463 CheckScriptFailure(lines, 'E1013: Argument 1: type mismatch, expected string but got number')
Bram Moolenaar65b95452020-07-19 14:03:09 +02001464enddef
1465
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001466def Test_vim9script_call_fail_const()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001467 var lines =<< trim END
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001468 vim9script
1469 const var = ''
1470 def MyFunc(arg: string)
1471 var = 'asdf'
1472 enddef
Bram Moolenaar822ba242020-05-24 23:00:18 +02001473 defcompile
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001474 END
1475 writefile(lines, 'Xcall_const.vim')
Bram Moolenaar9bd5d872020-09-06 21:47:48 +02001476 assert_fails('source Xcall_const.vim', 'E46:', '', 1, 'MyFunc')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001477 delete('Xcall_const.vim')
Bram Moolenaar3bdc90b2020-12-22 20:35:40 +01001478
1479 lines =<< trim END
1480 const g:Aconst = 77
1481 def Change()
1482 # comment
1483 g:Aconst = 99
1484 enddef
1485 call Change()
1486 unlet g:Aconst
1487 END
Bram Moolenaar1dcf55d2020-12-22 22:07:30 +01001488 CheckScriptFailure(lines, 'E741: Value is locked: Aconst', 2)
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001489enddef
1490
1491" Test that inside :function a Python function can be defined, :def is not
1492" recognized.
1493func Test_function_python()
1494 CheckFeature python3
Bram Moolenaar727345e2020-09-27 23:33:59 +02001495 let py = 'python3'
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001496 execute py "<< EOF"
1497def do_something():
1498 return 1
1499EOF
1500endfunc
1501
1502def Test_delfunc()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001503 var lines =<< trim END
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001504 vim9script
Bram Moolenaar4c17ad92020-04-27 22:47:51 +02001505 def g:GoneSoon()
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001506 echo 'hello'
1507 enddef
1508
1509 def CallGoneSoon()
1510 GoneSoon()
1511 enddef
Bram Moolenaar822ba242020-05-24 23:00:18 +02001512 defcompile
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001513
Bram Moolenaar4c17ad92020-04-27 22:47:51 +02001514 delfunc g:GoneSoon
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001515 CallGoneSoon()
1516 END
1517 writefile(lines, 'XToDelFunc')
Bram Moolenaar9bd5d872020-09-06 21:47:48 +02001518 assert_fails('so XToDelFunc', 'E933:', '', 1, 'CallGoneSoon')
1519 assert_fails('so XToDelFunc', 'E933:', '', 1, 'CallGoneSoon')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001520
1521 delete('XToDelFunc')
1522enddef
1523
1524def Test_redef_failure()
Bram Moolenaard2c61702020-09-06 15:58:36 +02001525 writefile(['def Func0(): string', 'return "Func0"', 'enddef'], 'Xdef')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001526 so Xdef
Bram Moolenaard2c61702020-09-06 15:58:36 +02001527 writefile(['def Func1(): string', 'return "Func1"', 'enddef'], 'Xdef')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001528 so Xdef
Bram Moolenaard2c61702020-09-06 15:58:36 +02001529 writefile(['def! Func0(): string', 'enddef', 'defcompile'], 'Xdef')
Bram Moolenaar9bd5d872020-09-06 21:47:48 +02001530 assert_fails('so Xdef', 'E1027:', '', 1, 'Func0')
Bram Moolenaard2c61702020-09-06 15:58:36 +02001531 writefile(['def Func2(): string', 'return "Func2"', 'enddef'], 'Xdef')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001532 so Xdef
Bram Moolenaard2c61702020-09-06 15:58:36 +02001533 delete('Xdef')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001534
Bram Moolenaar701cc6c2021-04-10 13:33:48 +02001535 assert_fails('g:Func0()', 'E1091:')
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001536 g:Func1()->assert_equal('Func1')
1537 g:Func2()->assert_equal('Func2')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001538
1539 delfunc! Func0
1540 delfunc! Func1
1541 delfunc! Func2
1542enddef
1543
Bram Moolenaarf93c7fe2020-04-23 22:16:53 +02001544def Test_vim9script_func()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001545 var lines =<< trim END
Bram Moolenaarf93c7fe2020-04-23 22:16:53 +02001546 vim9script
1547 func Func(arg)
1548 echo a:arg
1549 endfunc
1550 Func('text')
1551 END
1552 writefile(lines, 'XVim9Func')
1553 so XVim9Func
1554
1555 delete('XVim9Func')
1556enddef
1557
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001558let s:funcResult = 0
1559
1560def FuncNoArgNoRet()
Bram Moolenaar53900992020-08-22 19:02:02 +02001561 s:funcResult = 11
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001562enddef
1563
1564def FuncNoArgRetNumber(): number
Bram Moolenaar53900992020-08-22 19:02:02 +02001565 s:funcResult = 22
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001566 return 1234
1567enddef
1568
Bram Moolenaarec5929d2020-04-07 20:53:39 +02001569def FuncNoArgRetString(): string
Bram Moolenaar53900992020-08-22 19:02:02 +02001570 s:funcResult = 45
Bram Moolenaarec5929d2020-04-07 20:53:39 +02001571 return 'text'
1572enddef
1573
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001574def FuncOneArgNoRet(arg: number)
Bram Moolenaar53900992020-08-22 19:02:02 +02001575 s:funcResult = arg
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001576enddef
1577
1578def FuncOneArgRetNumber(arg: number): number
Bram Moolenaar53900992020-08-22 19:02:02 +02001579 s:funcResult = arg
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001580 return arg
1581enddef
1582
Bram Moolenaar08938ee2020-04-11 23:17:17 +02001583def FuncTwoArgNoRet(one: bool, two: number)
Bram Moolenaar53900992020-08-22 19:02:02 +02001584 s:funcResult = two
Bram Moolenaar08938ee2020-04-11 23:17:17 +02001585enddef
1586
Bram Moolenaarec5929d2020-04-07 20:53:39 +02001587def FuncOneArgRetString(arg: string): string
1588 return arg
1589enddef
1590
Bram Moolenaar89228602020-04-05 22:14:54 +02001591def FuncOneArgRetAny(arg: any): any
1592 return arg
1593enddef
1594
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001595def Test_func_type()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001596 var Ref1: func()
Bram Moolenaar53900992020-08-22 19:02:02 +02001597 s:funcResult = 0
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001598 Ref1 = FuncNoArgNoRet
1599 Ref1()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001600 s:funcResult->assert_equal(11)
Bram Moolenaar4c683752020-04-05 21:38:23 +02001601
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001602 var Ref2: func
Bram Moolenaar53900992020-08-22 19:02:02 +02001603 s:funcResult = 0
Bram Moolenaar4c683752020-04-05 21:38:23 +02001604 Ref2 = FuncNoArgNoRet
1605 Ref2()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001606 s:funcResult->assert_equal(11)
Bram Moolenaar4c683752020-04-05 21:38:23 +02001607
Bram Moolenaar53900992020-08-22 19:02:02 +02001608 s:funcResult = 0
Bram Moolenaar4c683752020-04-05 21:38:23 +02001609 Ref2 = FuncOneArgNoRet
1610 Ref2(12)
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001611 s:funcResult->assert_equal(12)
Bram Moolenaar4c683752020-04-05 21:38:23 +02001612
Bram Moolenaar53900992020-08-22 19:02:02 +02001613 s:funcResult = 0
Bram Moolenaar4c683752020-04-05 21:38:23 +02001614 Ref2 = FuncNoArgRetNumber
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001615 Ref2()->assert_equal(1234)
1616 s:funcResult->assert_equal(22)
Bram Moolenaar4c683752020-04-05 21:38:23 +02001617
Bram Moolenaar53900992020-08-22 19:02:02 +02001618 s:funcResult = 0
Bram Moolenaar4c683752020-04-05 21:38:23 +02001619 Ref2 = FuncOneArgRetNumber
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001620 Ref2(13)->assert_equal(13)
1621 s:funcResult->assert_equal(13)
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001622enddef
1623
Bram Moolenaar9978d472020-07-05 16:01:56 +02001624def Test_repeat_return_type()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001625 var res = 0
Bram Moolenaar9978d472020-07-05 16:01:56 +02001626 for n in repeat([1], 3)
1627 res += n
1628 endfor
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001629 res->assert_equal(3)
Bram Moolenaarfce82b32020-07-05 16:07:21 +02001630
1631 res = 0
1632 for n in add([1, 2], 3)
1633 res += n
1634 endfor
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001635 res->assert_equal(6)
Bram Moolenaar9978d472020-07-05 16:01:56 +02001636enddef
1637
Bram Moolenaar846178a2020-07-05 17:04:13 +02001638def Test_argv_return_type()
1639 next fileone filetwo
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001640 var res = ''
Bram Moolenaar846178a2020-07-05 17:04:13 +02001641 for name in argv()
1642 res ..= name
1643 endfor
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001644 res->assert_equal('fileonefiletwo')
Bram Moolenaar846178a2020-07-05 17:04:13 +02001645enddef
1646
Bram Moolenaarec5929d2020-04-07 20:53:39 +02001647def Test_func_type_part()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001648 var RefVoid: func: void
Bram Moolenaarec5929d2020-04-07 20:53:39 +02001649 RefVoid = FuncNoArgNoRet
1650 RefVoid = FuncOneArgNoRet
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001651 CheckDefFailure(['var RefVoid: func: void', 'RefVoid = FuncNoArgRetNumber'], 'E1012: Type mismatch; expected func(...) but got func(): number')
1652 CheckDefFailure(['var RefVoid: func: void', 'RefVoid = FuncNoArgRetString'], 'E1012: Type mismatch; expected func(...) but got func(): string')
Bram Moolenaarec5929d2020-04-07 20:53:39 +02001653
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001654 var RefAny: func(): any
Bram Moolenaarec5929d2020-04-07 20:53:39 +02001655 RefAny = FuncNoArgRetNumber
1656 RefAny = FuncNoArgRetString
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001657 CheckDefFailure(['var RefAny: func(): any', 'RefAny = FuncNoArgNoRet'], 'E1012: Type mismatch; expected func(): any but got func()')
1658 CheckDefFailure(['var RefAny: func(): any', 'RefAny = FuncOneArgNoRet'], 'E1012: Type mismatch; expected func(): any but got func(number)')
Bram Moolenaarec5929d2020-04-07 20:53:39 +02001659
Bram Moolenaar6abd3dc2020-10-04 14:17:32 +02001660 var RefAnyNoArgs: func: any = RefAny
1661
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001662 var RefNr: func: number
Bram Moolenaarec5929d2020-04-07 20:53:39 +02001663 RefNr = FuncNoArgRetNumber
1664 RefNr = FuncOneArgRetNumber
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001665 CheckDefFailure(['var RefNr: func: number', 'RefNr = FuncNoArgNoRet'], 'E1012: Type mismatch; expected func(...): number but got func()')
1666 CheckDefFailure(['var RefNr: func: number', 'RefNr = FuncNoArgRetString'], 'E1012: Type mismatch; expected func(...): number but got func(): string')
Bram Moolenaarec5929d2020-04-07 20:53:39 +02001667
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001668 var RefStr: func: string
Bram Moolenaarec5929d2020-04-07 20:53:39 +02001669 RefStr = FuncNoArgRetString
1670 RefStr = FuncOneArgRetString
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001671 CheckDefFailure(['var RefStr: func: string', 'RefStr = FuncNoArgNoRet'], 'E1012: Type mismatch; expected func(...): string but got func()')
1672 CheckDefFailure(['var RefStr: func: string', 'RefStr = FuncNoArgRetNumber'], 'E1012: Type mismatch; expected func(...): string but got func(): number')
Bram Moolenaarec5929d2020-04-07 20:53:39 +02001673enddef
1674
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001675def Test_func_type_fails()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001676 CheckDefFailure(['var ref1: func()'], 'E704:')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001677
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001678 CheckDefFailure(['var Ref1: func()', 'Ref1 = FuncNoArgRetNumber'], 'E1012: Type mismatch; expected func() but got func(): number')
1679 CheckDefFailure(['var Ref1: func()', 'Ref1 = FuncOneArgNoRet'], 'E1012: Type mismatch; expected func() but got func(number)')
1680 CheckDefFailure(['var Ref1: func()', 'Ref1 = FuncOneArgRetNumber'], 'E1012: Type mismatch; expected func() but got func(number): number')
1681 CheckDefFailure(['var Ref1: func(bool)', 'Ref1 = FuncTwoArgNoRet'], 'E1012: Type mismatch; expected func(bool) but got func(bool, number)')
1682 CheckDefFailure(['var Ref1: func(?bool)', 'Ref1 = FuncTwoArgNoRet'], 'E1012: Type mismatch; expected func(?bool) but got func(bool, number)')
1683 CheckDefFailure(['var Ref1: func(...bool)', 'Ref1 = FuncTwoArgNoRet'], 'E1012: Type mismatch; expected func(...bool) but got func(bool, number)')
Bram Moolenaar08938ee2020-04-11 23:17:17 +02001684
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001685 CheckDefFailure(['var RefWrong: func(string ,number)'], 'E1068:')
1686 CheckDefFailure(['var RefWrong: func(string,number)'], 'E1069:')
1687 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:')
1688 CheckDefFailure(['var RefWrong: func(bool):string'], 'E1069:')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001689enddef
1690
Bram Moolenaar89228602020-04-05 22:14:54 +02001691def Test_func_return_type()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001692 var nr: number
Bram Moolenaar89228602020-04-05 22:14:54 +02001693 nr = FuncNoArgRetNumber()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001694 nr->assert_equal(1234)
Bram Moolenaar89228602020-04-05 22:14:54 +02001695
1696 nr = FuncOneArgRetAny(122)
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001697 nr->assert_equal(122)
Bram Moolenaar89228602020-04-05 22:14:54 +02001698
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001699 var str: string
Bram Moolenaar89228602020-04-05 22:14:54 +02001700 str = FuncOneArgRetAny('yes')
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001701 str->assert_equal('yes')
Bram Moolenaar89228602020-04-05 22:14:54 +02001702
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001703 CheckDefFailure(['var str: string', 'str = FuncNoArgRetNumber()'], 'E1012: Type mismatch; expected string but got number')
Bram Moolenaar89228602020-04-05 22:14:54 +02001704enddef
1705
Bram Moolenaar6abd3dc2020-10-04 14:17:32 +02001706def Test_func_common_type()
1707 def FuncOne(n: number): number
1708 return n
1709 enddef
1710 def FuncTwo(s: string): number
1711 return len(s)
1712 enddef
1713 def FuncThree(n: number, s: string): number
1714 return n + len(s)
1715 enddef
1716 var list = [FuncOne, FuncTwo, FuncThree]
1717 assert_equal(8, list[0](8))
1718 assert_equal(4, list[1]('word'))
1719 assert_equal(7, list[2](3, 'word'))
1720enddef
1721
Bram Moolenaar5e774c72020-04-12 21:53:00 +02001722def MultiLine(
1723 arg1: string,
1724 arg2 = 1234,
1725 ...rest: list<string>
1726 ): string
1727 return arg1 .. arg2 .. join(rest, '-')
1728enddef
1729
Bram Moolenaar2c330432020-04-13 14:41:35 +02001730def MultiLineComment(
1731 arg1: string, # comment
1732 arg2 = 1234, # comment
1733 ...rest: list<string> # comment
1734 ): string # comment
1735 return arg1 .. arg2 .. join(rest, '-')
1736enddef
1737
Bram Moolenaar5e774c72020-04-12 21:53:00 +02001738def Test_multiline()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001739 MultiLine('text')->assert_equal('text1234')
1740 MultiLine('text', 777)->assert_equal('text777')
1741 MultiLine('text', 777, 'one')->assert_equal('text777one')
1742 MultiLine('text', 777, 'one', 'two')->assert_equal('text777one-two')
Bram Moolenaar5e774c72020-04-12 21:53:00 +02001743enddef
1744
Bram Moolenaar23e03252020-04-12 22:22:31 +02001745func Test_multiline_not_vim9()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001746 call MultiLine('text')->assert_equal('text1234')
1747 call MultiLine('text', 777)->assert_equal('text777')
1748 call MultiLine('text', 777, 'one')->assert_equal('text777one')
1749 call MultiLine('text', 777, 'one', 'two')->assert_equal('text777one-two')
Bram Moolenaar23e03252020-04-12 22:22:31 +02001750endfunc
1751
Bram Moolenaar5e774c72020-04-12 21:53:00 +02001752
Bram Moolenaaree4e0c12020-04-06 21:35:05 +02001753" When using CheckScriptFailure() for the below test, E1010 is generated instead
1754" of E1056.
1755func Test_E1056_1059()
1756 let caught_1056 = 0
1757 try
1758 def F():
1759 return 1
1760 enddef
1761 catch /E1056:/
1762 let caught_1056 = 1
1763 endtry
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001764 eval caught_1056->assert_equal(1)
Bram Moolenaaree4e0c12020-04-06 21:35:05 +02001765
1766 let caught_1059 = 0
1767 try
1768 def F5(items : list)
1769 echo 'a'
1770 enddef
1771 catch /E1059:/
1772 let caught_1059 = 1
1773 endtry
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001774 eval caught_1059->assert_equal(1)
Bram Moolenaaree4e0c12020-04-06 21:35:05 +02001775endfunc
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001776
Bram Moolenaar015f4262020-05-05 21:25:22 +02001777func DelMe()
1778 echo 'DelMe'
1779endfunc
1780
Bram Moolenaarbf8feb52020-08-08 14:26:31 +02001781def Test_error_reporting()
1782 # comment lines at the start of the function
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001783 var lines =<< trim END
Bram Moolenaarbf8feb52020-08-08 14:26:31 +02001784 " comment
1785 def Func()
1786 # comment
1787 # comment
1788 invalid
1789 enddef
1790 defcompile
1791 END
Bram Moolenaar08052222020-09-14 17:04:31 +02001792 writefile(lines, 'Xdef')
Bram Moolenaarbf8feb52020-08-08 14:26:31 +02001793 try
1794 source Xdef
Bram Moolenaar7517ffd2020-08-14 18:35:07 +02001795 assert_report('should have failed')
Bram Moolenaarbf8feb52020-08-08 14:26:31 +02001796 catch /E476:/
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001797 v:exception->assert_match('Invalid command: invalid')
1798 v:throwpoint->assert_match(', line 3$')
Bram Moolenaarbf8feb52020-08-08 14:26:31 +02001799 endtry
Bram Moolenaar2d870f82020-12-05 13:41:01 +01001800 delfunc! g:Func
Bram Moolenaarbf8feb52020-08-08 14:26:31 +02001801
1802 # comment lines after the start of the function
1803 lines =<< trim END
1804 " comment
1805 def Func()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001806 var x = 1234
Bram Moolenaarbf8feb52020-08-08 14:26:31 +02001807 # comment
1808 # comment
1809 invalid
1810 enddef
1811 defcompile
1812 END
Bram Moolenaar08052222020-09-14 17:04:31 +02001813 writefile(lines, 'Xdef')
Bram Moolenaarbf8feb52020-08-08 14:26:31 +02001814 try
1815 source Xdef
Bram Moolenaar7517ffd2020-08-14 18:35:07 +02001816 assert_report('should have failed')
Bram Moolenaarbf8feb52020-08-08 14:26:31 +02001817 catch /E476:/
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001818 v:exception->assert_match('Invalid command: invalid')
1819 v:throwpoint->assert_match(', line 4$')
Bram Moolenaarbf8feb52020-08-08 14:26:31 +02001820 endtry
Bram Moolenaar2d870f82020-12-05 13:41:01 +01001821 delfunc! g:Func
Bram Moolenaarbf8feb52020-08-08 14:26:31 +02001822
Bram Moolenaar7517ffd2020-08-14 18:35:07 +02001823 lines =<< trim END
1824 vim9script
1825 def Func()
Bram Moolenaare0de1712020-12-02 17:36:54 +01001826 var db = {foo: 1, bar: 2}
Bram Moolenaar7517ffd2020-08-14 18:35:07 +02001827 # comment
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001828 var x = db.asdf
Bram Moolenaar7517ffd2020-08-14 18:35:07 +02001829 enddef
1830 defcompile
1831 Func()
1832 END
Bram Moolenaar08052222020-09-14 17:04:31 +02001833 writefile(lines, 'Xdef')
Bram Moolenaar7517ffd2020-08-14 18:35:07 +02001834 try
1835 source Xdef
1836 assert_report('should have failed')
1837 catch /E716:/
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001838 v:throwpoint->assert_match('_Func, line 3$')
Bram Moolenaar7517ffd2020-08-14 18:35:07 +02001839 endtry
Bram Moolenaar2d870f82020-12-05 13:41:01 +01001840 delfunc! g:Func
Bram Moolenaar7517ffd2020-08-14 18:35:07 +02001841
Bram Moolenaar08052222020-09-14 17:04:31 +02001842 delete('Xdef')
Bram Moolenaarbf8feb52020-08-08 14:26:31 +02001843enddef
1844
Bram Moolenaar015f4262020-05-05 21:25:22 +02001845def Test_deleted_function()
1846 CheckDefExecFailure([
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001847 'var RefMe: func = function("g:DelMe")',
Bram Moolenaar015f4262020-05-05 21:25:22 +02001848 'delfunc g:DelMe',
1849 'echo RefMe()'], 'E117:')
1850enddef
1851
1852def Test_unknown_function()
1853 CheckDefExecFailure([
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001854 'var Ref: func = function("NotExist")',
Bram Moolenaar9b7bf9e2020-07-11 22:14:59 +02001855 'delfunc g:NotExist'], 'E700:')
Bram Moolenaar015f4262020-05-05 21:25:22 +02001856enddef
1857
Bram Moolenaar328eac22021-01-07 19:23:08 +01001858def RefFunc(Ref: func(any): any): string
Bram Moolenaarc8cd2b32020-05-01 19:29:08 +02001859 return Ref('more')
1860enddef
1861
1862def Test_closure_simple()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001863 var local = 'some '
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01001864 RefFunc((s) => local .. s)->assert_equal('some more')
Bram Moolenaarc8cd2b32020-05-01 19:29:08 +02001865enddef
1866
Bram Moolenaarbf67ea12020-05-02 17:52:42 +02001867def MakeRef()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001868 var local = 'some '
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01001869 g:Ref = (s) => local .. s
Bram Moolenaarbf67ea12020-05-02 17:52:42 +02001870enddef
1871
1872def Test_closure_ref_after_return()
1873 MakeRef()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001874 g:Ref('thing')->assert_equal('some thing')
Bram Moolenaarbf67ea12020-05-02 17:52:42 +02001875 unlet g:Ref
1876enddef
1877
Bram Moolenaar5adc55c2020-05-02 23:12:58 +02001878def MakeTwoRefs()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001879 var local = ['some']
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01001880 g:Extend = (s) => local->add(s)
1881 g:Read = () => local
Bram Moolenaar5adc55c2020-05-02 23:12:58 +02001882enddef
1883
1884def Test_closure_two_refs()
1885 MakeTwoRefs()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001886 join(g:Read(), ' ')->assert_equal('some')
Bram Moolenaar5adc55c2020-05-02 23:12:58 +02001887 g:Extend('more')
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001888 join(g:Read(), ' ')->assert_equal('some more')
Bram Moolenaar5adc55c2020-05-02 23:12:58 +02001889 g:Extend('even')
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001890 join(g:Read(), ' ')->assert_equal('some more even')
Bram Moolenaar5adc55c2020-05-02 23:12:58 +02001891
1892 unlet g:Extend
1893 unlet g:Read
1894enddef
1895
Bram Moolenaar5adc55c2020-05-02 23:12:58 +02001896def ReadRef(Ref: func(): list<string>): string
1897 return join(Ref(), ' ')
1898enddef
1899
Bram Moolenaar5e654232020-09-16 15:22:00 +02001900def ExtendRef(Ref: func(string): list<string>, add: string)
Bram Moolenaar5adc55c2020-05-02 23:12:58 +02001901 Ref(add)
1902enddef
1903
1904def Test_closure_two_indirect_refs()
Bram Moolenaarf7779c62020-05-03 15:38:16 +02001905 MakeTwoRefs()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001906 ReadRef(g:Read)->assert_equal('some')
Bram Moolenaar5adc55c2020-05-02 23:12:58 +02001907 ExtendRef(g:Extend, 'more')
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001908 ReadRef(g:Read)->assert_equal('some more')
Bram Moolenaar5adc55c2020-05-02 23:12:58 +02001909 ExtendRef(g:Extend, 'even')
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001910 ReadRef(g:Read)->assert_equal('some more even')
Bram Moolenaar5adc55c2020-05-02 23:12:58 +02001911
1912 unlet g:Extend
1913 unlet g:Read
1914enddef
Bram Moolenaarbf67ea12020-05-02 17:52:42 +02001915
Bram Moolenaar2fd4cd72020-05-03 22:30:49 +02001916def MakeArgRefs(theArg: string)
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001917 var local = 'loc_val'
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01001918 g:UseArg = (s) => theArg .. '/' .. local .. '/' .. s
Bram Moolenaar2fd4cd72020-05-03 22:30:49 +02001919enddef
1920
1921def MakeArgRefsVarargs(theArg: string, ...rest: list<string>)
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001922 var local = 'the_loc'
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01001923 g:UseVararg = (s) => theArg .. '/' .. local .. '/' .. s .. '/' .. join(rest)
Bram Moolenaar2fd4cd72020-05-03 22:30:49 +02001924enddef
1925
1926def Test_closure_using_argument()
1927 MakeArgRefs('arg_val')
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001928 g:UseArg('call_val')->assert_equal('arg_val/loc_val/call_val')
Bram Moolenaar2fd4cd72020-05-03 22:30:49 +02001929
1930 MakeArgRefsVarargs('arg_val', 'one', 'two')
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001931 g:UseVararg('call_val')->assert_equal('arg_val/the_loc/call_val/one two')
Bram Moolenaar2fd4cd72020-05-03 22:30:49 +02001932
1933 unlet g:UseArg
1934 unlet g:UseVararg
Bram Moolenaar44ec21c2021-02-12 21:50:57 +01001935
1936 var lines =<< trim END
1937 vim9script
1938 def Test(Fun: func(number): number): list<number>
1939 return map([1, 2, 3], (_, i) => Fun(i))
1940 enddef
1941 def Inc(nr: number): number
1942 return nr + 2
1943 enddef
1944 assert_equal([3, 4, 5], Test(Inc))
1945 END
1946 CheckScriptSuccess(lines)
Bram Moolenaar2fd4cd72020-05-03 22:30:49 +02001947enddef
1948
Bram Moolenaar85d5e2b2020-10-10 14:13:01 +02001949def MakeGetAndAppendRefs()
1950 var local = 'a'
1951
1952 def Append(arg: string)
1953 local ..= arg
1954 enddef
1955 g:Append = Append
1956
1957 def Get(): string
1958 return local
1959 enddef
1960 g:Get = Get
1961enddef
1962
1963def Test_closure_append_get()
1964 MakeGetAndAppendRefs()
1965 g:Get()->assert_equal('a')
1966 g:Append('-b')
1967 g:Get()->assert_equal('a-b')
1968 g:Append('-c')
1969 g:Get()->assert_equal('a-b-c')
1970
1971 unlet g:Append
1972 unlet g:Get
1973enddef
Bram Moolenaarb68b3462020-05-06 21:06:30 +02001974
Bram Moolenaar04b12692020-05-04 23:24:44 +02001975def Test_nested_closure()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001976 var local = 'text'
Bram Moolenaar04b12692020-05-04 23:24:44 +02001977 def Closure(arg: string): string
1978 return local .. arg
1979 enddef
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001980 Closure('!!!')->assert_equal('text!!!')
Bram Moolenaar04b12692020-05-04 23:24:44 +02001981enddef
1982
Bram Moolenaar6f5b6df2020-05-16 21:20:12 +02001983func GetResult(Ref)
1984 return a:Ref('some')
1985endfunc
1986
1987def Test_call_closure_not_compiled()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001988 var text = 'text'
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01001989 g:Ref = (s) => s .. text
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001990 GetResult(g:Ref)->assert_equal('sometext')
Bram Moolenaar6f5b6df2020-05-16 21:20:12 +02001991enddef
1992
Bram Moolenaar7cbfaa52020-09-18 21:25:32 +02001993def Test_double_closure_fails()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001994 var lines =<< trim END
Bram Moolenaar7cbfaa52020-09-18 21:25:32 +02001995 vim9script
1996 def Func()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001997 var name = 0
1998 for i in range(2)
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01001999 timer_start(0, () => name)
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002000 endfor
Bram Moolenaar7cbfaa52020-09-18 21:25:32 +02002001 enddef
2002 Func()
2003 END
Bram Moolenaar148ce7a2020-09-23 21:57:23 +02002004 CheckScriptSuccess(lines)
Bram Moolenaar7cbfaa52020-09-18 21:25:32 +02002005enddef
2006
Bram Moolenaar85d5e2b2020-10-10 14:13:01 +02002007def Test_nested_closure_used()
2008 var lines =<< trim END
2009 vim9script
2010 def Func()
2011 var x = 'hello'
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01002012 var Closure = () => x
2013 g:Myclosure = () => Closure()
Bram Moolenaar85d5e2b2020-10-10 14:13:01 +02002014 enddef
2015 Func()
2016 assert_equal('hello', g:Myclosure())
2017 END
2018 CheckScriptSuccess(lines)
2019enddef
Bram Moolenaar0876c782020-10-07 19:08:04 +02002020
Bram Moolenaarc70bdab2020-09-26 19:59:38 +02002021def Test_nested_closure_fails()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002022 var lines =<< trim END
Bram Moolenaarc70bdab2020-09-26 19:59:38 +02002023 vim9script
2024 def FuncA()
2025 FuncB(0)
2026 enddef
2027 def FuncB(n: number): list<string>
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01002028 return map([0], (_, v) => n)
Bram Moolenaarc70bdab2020-09-26 19:59:38 +02002029 enddef
2030 FuncA()
2031 END
2032 CheckScriptFailure(lines, 'E1012:')
2033enddef
2034
Bram Moolenaarf112f302020-12-20 17:47:52 +01002035def Test_global_closure()
2036 var lines =<< trim END
2037 vim9script
2038 def ReverseEveryNLines(n: number, line1: number, line2: number)
2039 var mods = 'sil keepj keepp lockm '
2040 var range = ':' .. line1 .. ',' .. line2
2041 def g:Offset(): number
2042 var offset = (line('.') - line1 + 1) % n
2043 return offset != 0 ? offset : n
2044 enddef
2045 exe mods .. range .. 'g/^/exe "m .-" .. g:Offset()'
2046 enddef
2047
2048 new
2049 repeat(['aaa', 'bbb', 'ccc'], 3)->setline(1)
2050 ReverseEveryNLines(3, 1, 9)
2051 END
2052 CheckScriptSuccess(lines)
2053 var expected = repeat(['ccc', 'bbb', 'aaa'], 3)
2054 assert_equal(expected, getline(1, 9))
2055 bwipe!
2056enddef
2057
Bram Moolenaarcd45ed02020-12-22 17:35:54 +01002058def Test_global_closure_called_directly()
2059 var lines =<< trim END
2060 vim9script
2061 def Outer()
2062 var x = 1
2063 def g:Inner()
2064 var y = x
2065 x += 1
2066 assert_equal(1, y)
2067 enddef
2068 g:Inner()
2069 assert_equal(2, x)
2070 enddef
2071 Outer()
2072 END
2073 CheckScriptSuccess(lines)
2074 delfunc g:Inner
2075enddef
2076
Bram Moolenaar34c54eb2020-11-25 19:15:19 +01002077def Test_failure_in_called_function()
2078 # this was using the frame index as the return value
2079 var lines =<< trim END
2080 vim9script
2081 au TerminalWinOpen * eval [][0]
2082 def PopupTerm(a: any)
2083 # make sure typvals on stack are string
2084 ['a', 'b', 'c', 'd', 'e', 'f', 'g']->join()
2085 FireEvent()
2086 enddef
2087 def FireEvent()
2088 do TerminalWinOpen
2089 enddef
2090 # use try/catch to make eval fail
2091 try
2092 call PopupTerm(0)
2093 catch
2094 endtry
2095 au! TerminalWinOpen
2096 END
2097 CheckScriptSuccess(lines)
2098enddef
2099
Bram Moolenaar5366e1a2020-10-01 13:01:34 +02002100def Test_nested_lambda()
2101 var lines =<< trim END
2102 vim9script
2103 def Func()
2104 var x = 4
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01002105 var Lambda1 = () => 7
2106 var Lambda2 = () => [Lambda1(), x]
Bram Moolenaar5366e1a2020-10-01 13:01:34 +02002107 var res = Lambda2()
2108 assert_equal([7, 4], res)
2109 enddef
2110 Func()
2111 END
2112 CheckScriptSuccess(lines)
2113enddef
2114
Bram Moolenaar074f84c2021-05-18 11:47:44 +02002115def Test_nested_inline_lambda()
2116 # TODO: use the "text" argument
2117 var lines =<< trim END
2118 vim9script
2119 def F(text: string): func(string): func(string): string
2120 return (arg: string): func(string): string => ((sep: string): string => {
2121 return sep .. arg
2122 })
2123 enddef
2124 assert_equal('--there', F('unused')('there')('--'))
2125 END
2126 CheckScriptSuccess(lines)
2127enddef
2128
Bram Moolenaar52bf81c2020-11-17 18:50:44 +01002129def Shadowed(): list<number>
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01002130 var FuncList: list<func: number> = [() => 42]
Bram Moolenaar75ab91f2021-01-10 22:42:50 +01002131 return FuncList->mapnew((_, Shadowed) => Shadowed())
Bram Moolenaar52bf81c2020-11-17 18:50:44 +01002132enddef
2133
2134def Test_lambda_arg_shadows_func()
2135 assert_equal([42], Shadowed())
2136enddef
2137
Bram Moolenaaracd4c5e2020-06-22 19:39:03 +02002138def Line_continuation_in_def(dir: string = ''): string
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002139 var path: string = empty(dir)
2140 \ ? 'empty'
2141 \ : 'full'
2142 return path
Bram Moolenaaracd4c5e2020-06-22 19:39:03 +02002143enddef
2144
2145def Test_line_continuation_in_def()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002146 Line_continuation_in_def('.')->assert_equal('full')
Bram Moolenaaracd4c5e2020-06-22 19:39:03 +02002147enddef
2148
Bram Moolenaar2ea95b62020-11-19 21:47:56 +01002149def Test_script_var_in_lambda()
2150 var lines =<< trim END
2151 vim9script
2152 var script = 'test'
Bram Moolenaarbb8a7ce2021-04-10 20:10:26 +02002153 assert_equal(['test'], map(['one'], (_, _) => script))
Bram Moolenaar2ea95b62020-11-19 21:47:56 +01002154 END
2155 CheckScriptSuccess(lines)
2156enddef
2157
Bram Moolenaar5e654232020-09-16 15:22:00 +02002158def Line_continuation_in_lambda(): list<string>
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002159 var x = range(97, 100)
Bram Moolenaar75ab91f2021-01-10 22:42:50 +01002160 ->mapnew((_, v) => nr2char(v)
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01002161 ->toupper())
Bram Moolenaar7a4b8982020-07-08 17:36:21 +02002162 ->reverse()
2163 return x
2164enddef
2165
2166def Test_line_continuation_in_lambda()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002167 Line_continuation_in_lambda()->assert_equal(['D', 'C', 'B', 'A'])
Bram Moolenaarf898f7c2021-01-16 18:09:52 +01002168
2169 var lines =<< trim END
2170 vim9script
2171 var res = [{n: 1, m: 2, s: 'xxx'}]
2172 ->mapnew((_, v: dict<any>): string => printf('%d:%d:%s',
2173 v.n,
2174 v.m,
2175 substitute(v.s, '.*', 'yyy', '')
2176 ))
2177 assert_equal(['1:2:yyy'], res)
2178 END
2179 CheckScriptSuccess(lines)
Bram Moolenaar7a4b8982020-07-08 17:36:21 +02002180enddef
2181
Bram Moolenaarb6571982021-01-08 22:24:19 +01002182def Test_list_lambda()
2183 timer_start(1000, (_) => 0)
2184 var body = execute(timer_info()[0].callback
2185 ->string()
2186 ->substitute("('", ' ', '')
2187 ->substitute("')", '', '')
2188 ->substitute('function\zs', ' ', ''))
Bram Moolenaar767034c2021-04-09 17:24:52 +02002189 assert_match('def <lambda>\d\+(_: any): number\n1 return 0\n enddef', body)
Bram Moolenaarb6571982021-01-08 22:24:19 +01002190enddef
2191
Bram Moolenaar96cf4ba2021-04-24 14:15:41 +02002192def Test_legacy_lambda()
2193 legacy echo {x -> 'hello ' .. x}('foo')
Bram Moolenaardc4c2302021-04-25 13:54:42 +02002194
Bram Moolenaar96cf4ba2021-04-24 14:15:41 +02002195 var lines =<< trim END
2196 echo {x -> 'hello ' .. x}('foo')
2197 END
2198 CheckDefAndScriptFailure(lines, 'E720:')
Bram Moolenaardc4c2302021-04-25 13:54:42 +02002199
2200 lines =<< trim END
2201 vim9script
2202 def Func()
2203 echo (() => 'no error')()
2204 enddef
2205 legacy call s:Func()
2206 END
2207 CheckScriptSuccess(lines)
Bram Moolenaar96cf4ba2021-04-24 14:15:41 +02002208enddef
2209
Bram Moolenaarc3cb1c92021-06-02 16:47:53 +02002210def Test_legacy_errors()
2211 for cmd in ['if', 'elseif', 'else', 'endif',
2212 'for', 'endfor', 'continue', 'break',
2213 'while', 'endwhile',
2214 'try', 'catch', 'finally', 'endtry']
2215 CheckDefFailure(['legacy ' .. cmd .. ' expr'], 'E1189:')
2216 endfor
2217enddef
2218
Bram Moolenaarab360522021-01-10 14:02:28 +01002219def DoFilterThis(a: string): list<string>
2220 # closure nested inside another closure using argument
2221 var Filter = (l) => filter(l, (_, v) => stridx(v, a) == 0)
2222 return ['x', 'y', 'a', 'x2', 'c']->Filter()
2223enddef
2224
2225def Test_nested_closure_using_argument()
2226 assert_equal(['x', 'x2'], DoFilterThis('x'))
2227enddef
2228
Bram Moolenaar0186e582021-01-10 18:33:11 +01002229def Test_triple_nested_closure()
2230 var what = 'x'
2231 var Match = (val: string, cmp: string): bool => stridx(val, cmp) == 0
2232 var Filter = (l) => filter(l, (_, v) => Match(v, what))
2233 assert_equal(['x', 'x2'], ['x', 'y', 'a', 'x2', 'c']->Filter())
2234enddef
2235
Bram Moolenaar8f510af2020-07-05 18:48:23 +02002236func Test_silent_echo()
Bram Moolenaar47e7d702020-07-05 18:18:42 +02002237 CheckScreendump
2238
2239 let lines =<< trim END
2240 vim9script
2241 def EchoNothing()
2242 silent echo ''
2243 enddef
2244 defcompile
2245 END
Bram Moolenaar8f510af2020-07-05 18:48:23 +02002246 call writefile(lines, 'XTest_silent_echo')
Bram Moolenaar47e7d702020-07-05 18:18:42 +02002247
2248 " Check that the balloon shows up after a mouse move
2249 let buf = RunVimInTerminal('-S XTest_silent_echo', {'rows': 6})
Bram Moolenaar8f510af2020-07-05 18:48:23 +02002250 call term_sendkeys(buf, ":abc")
Bram Moolenaar47e7d702020-07-05 18:18:42 +02002251 call VerifyScreenDump(buf, 'Test_vim9_silent_echo', {})
2252
2253 " clean up
2254 call StopVimInTerminal(buf)
2255 call delete('XTest_silent_echo')
Bram Moolenaar8f510af2020-07-05 18:48:23 +02002256endfunc
Bram Moolenaar47e7d702020-07-05 18:18:42 +02002257
Bram Moolenaar171fb922020-10-28 16:54:47 +01002258def SilentlyError()
2259 execute('silent! invalid')
2260 g:did_it = 'yes'
2261enddef
2262
Bram Moolenaar28ee8922020-10-28 20:20:00 +01002263func UserError()
2264 silent! invalid
2265endfunc
2266
2267def SilentlyUserError()
2268 UserError()
2269 g:did_it = 'yes'
2270enddef
Bram Moolenaar171fb922020-10-28 16:54:47 +01002271
2272" This can't be a :def function, because the assert would not be reached.
Bram Moolenaar171fb922020-10-28 16:54:47 +01002273func Test_ignore_silent_error()
2274 let g:did_it = 'no'
2275 call SilentlyError()
2276 call assert_equal('yes', g:did_it)
2277
Bram Moolenaar28ee8922020-10-28 20:20:00 +01002278 let g:did_it = 'no'
2279 call SilentlyUserError()
2280 call assert_equal('yes', g:did_it)
Bram Moolenaar171fb922020-10-28 16:54:47 +01002281
2282 unlet g:did_it
2283endfunc
2284
Bram Moolenaarcd030c42020-10-30 21:49:40 +01002285def Test_ignore_silent_error_in_filter()
2286 var lines =<< trim END
2287 vim9script
2288 def Filter(winid: number, key: string): bool
2289 if key == 'o'
2290 silent! eval [][0]
2291 return true
2292 endif
2293 return popup_filter_menu(winid, key)
2294 enddef
2295
Bram Moolenaare0de1712020-12-02 17:36:54 +01002296 popup_create('popup', {filter: Filter})
Bram Moolenaarcd030c42020-10-30 21:49:40 +01002297 feedkeys("o\r", 'xnt')
2298 END
2299 CheckScriptSuccess(lines)
2300enddef
2301
Bram Moolenaar4b9bd692020-09-05 21:57:53 +02002302def Fibonacci(n: number): number
2303 if n < 2
2304 return n
2305 else
2306 return Fibonacci(n - 1) + Fibonacci(n - 2)
2307 endif
2308enddef
2309
Bram Moolenaar985116a2020-07-12 17:31:09 +02002310def Test_recursive_call()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002311 Fibonacci(20)->assert_equal(6765)
Bram Moolenaar985116a2020-07-12 17:31:09 +02002312enddef
2313
Bram Moolenaar08f7a412020-07-13 20:41:08 +02002314def TreeWalk(dir: string): list<any>
Bram Moolenaar75ab91f2021-01-10 22:42:50 +01002315 return readdir(dir)->mapnew((_, val) =>
Bram Moolenaar08f7a412020-07-13 20:41:08 +02002316 fnamemodify(dir .. '/' .. val, ':p')->isdirectory()
Bram Moolenaar2bede172020-11-19 18:53:18 +01002317 ? {[val]: TreeWalk(dir .. '/' .. val)}
Bram Moolenaar08f7a412020-07-13 20:41:08 +02002318 : val
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01002319 )
Bram Moolenaar08f7a412020-07-13 20:41:08 +02002320enddef
2321
2322def Test_closure_in_map()
2323 mkdir('XclosureDir/tdir', 'p')
2324 writefile(['111'], 'XclosureDir/file1')
2325 writefile(['222'], 'XclosureDir/file2')
2326 writefile(['333'], 'XclosureDir/tdir/file3')
2327
Bram Moolenaare0de1712020-12-02 17:36:54 +01002328 TreeWalk('XclosureDir')->assert_equal(['file1', 'file2', {tdir: ['file3']}])
Bram Moolenaar08f7a412020-07-13 20:41:08 +02002329
2330 delete('XclosureDir', 'rf')
2331enddef
2332
Bram Moolenaar7b5d5442020-10-04 13:42:34 +02002333def Test_invalid_function_name()
2334 var lines =<< trim END
2335 vim9script
2336 def s: list<string>
2337 END
2338 CheckScriptFailure(lines, 'E129:')
2339
2340 lines =<< trim END
2341 vim9script
2342 def g: list<string>
2343 END
2344 CheckScriptFailure(lines, 'E129:')
2345
2346 lines =<< trim END
2347 vim9script
2348 def <SID>: list<string>
2349 END
2350 CheckScriptFailure(lines, 'E884:')
2351
2352 lines =<< trim END
2353 vim9script
2354 def F list<string>
2355 END
2356 CheckScriptFailure(lines, 'E488:')
2357enddef
2358
Bram Moolenaara90afb92020-07-15 22:38:56 +02002359def Test_partial_call()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002360 var Xsetlist = function('setloclist', [0])
Bram Moolenaare0de1712020-12-02 17:36:54 +01002361 Xsetlist([], ' ', {title: 'test'})
2362 getloclist(0, {title: 1})->assert_equal({title: 'test'})
Bram Moolenaara90afb92020-07-15 22:38:56 +02002363
2364 Xsetlist = function('setloclist', [0, [], ' '])
Bram Moolenaare0de1712020-12-02 17:36:54 +01002365 Xsetlist({title: 'test'})
2366 getloclist(0, {title: 1})->assert_equal({title: 'test'})
Bram Moolenaara90afb92020-07-15 22:38:56 +02002367
2368 Xsetlist = function('setqflist')
Bram Moolenaare0de1712020-12-02 17:36:54 +01002369 Xsetlist([], ' ', {title: 'test'})
2370 getqflist({title: 1})->assert_equal({title: 'test'})
Bram Moolenaara90afb92020-07-15 22:38:56 +02002371
2372 Xsetlist = function('setqflist', [[], ' '])
Bram Moolenaare0de1712020-12-02 17:36:54 +01002373 Xsetlist({title: 'test'})
2374 getqflist({title: 1})->assert_equal({title: 'test'})
Bram Moolenaar6abd3dc2020-10-04 14:17:32 +02002375
2376 var Len: func: number = function('len', ['word'])
2377 assert_equal(4, Len())
Bram Moolenaara90afb92020-07-15 22:38:56 +02002378enddef
2379
Bram Moolenaar2dd0a2c2020-08-08 15:10:27 +02002380def Test_cmd_modifier()
2381 tab echo '0'
Bram Moolenaard2c61702020-09-06 15:58:36 +02002382 CheckDefFailure(['5tab echo 3'], 'E16:')
Bram Moolenaar2dd0a2c2020-08-08 15:10:27 +02002383enddef
2384
2385def Test_restore_modifiers()
2386 # check that when compiling a :def function command modifiers are not messed
2387 # up.
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002388 var lines =<< trim END
Bram Moolenaar2dd0a2c2020-08-08 15:10:27 +02002389 vim9script
2390 set eventignore=
2391 autocmd QuickFixCmdPost * copen
2392 def AutocmdsDisabled()
Bram Moolenaar6cf7e3b2020-10-28 14:31:16 +01002393 eval 0
Bram Moolenaar2dd0a2c2020-08-08 15:10:27 +02002394 enddef
2395 func Func()
2396 noautocmd call s:AutocmdsDisabled()
2397 let g:ei_after = &eventignore
2398 endfunc
2399 Func()
2400 END
2401 CheckScriptSuccess(lines)
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002402 g:ei_after->assert_equal('')
Bram Moolenaar2dd0a2c2020-08-08 15:10:27 +02002403enddef
2404
Bram Moolenaardfa3d552020-09-10 22:05:08 +02002405def StackTop()
2406 eval 1
2407 eval 2
2408 # call not on fourth line
2409 StackBot()
2410enddef
2411
2412def StackBot()
2413 # throw an error
2414 eval [][0]
2415enddef
2416
2417def Test_callstack_def()
2418 try
2419 StackTop()
2420 catch
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002421 v:throwpoint->assert_match('Test_callstack_def\[2\]..StackTop\[4\]..StackBot, line 2')
Bram Moolenaardfa3d552020-09-10 22:05:08 +02002422 endtry
2423enddef
2424
Bram Moolenaare8211a32020-10-09 22:04:29 +02002425" Re-using spot for variable used in block
2426def Test_block_scoped_var()
2427 var lines =<< trim END
2428 vim9script
2429 def Func()
2430 var x = ['a', 'b', 'c']
2431 if 1
2432 var y = 'x'
Bram Moolenaarbb8a7ce2021-04-10 20:10:26 +02002433 map(x, (_, _) => y)
Bram Moolenaare8211a32020-10-09 22:04:29 +02002434 endif
2435 var z = x
2436 assert_equal(['x', 'x', 'x'], z)
2437 enddef
2438 Func()
2439 END
2440 CheckScriptSuccess(lines)
2441enddef
2442
Bram Moolenaareeece9e2020-11-20 19:26:48 +01002443def Test_reset_did_emsg()
2444 var lines =<< trim END
2445 @s = 'blah'
2446 au BufWinLeave * #
2447 def Func()
2448 var winid = popup_create('popup', {})
2449 exe '*s'
2450 popup_close(winid)
2451 enddef
2452 Func()
2453 END
2454 CheckScriptFailure(lines, 'E492:', 8)
Bram Moolenaar2d870f82020-12-05 13:41:01 +01002455 delfunc! g:Func
Bram Moolenaareeece9e2020-11-20 19:26:48 +01002456enddef
2457
Bram Moolenaar57f799e2020-12-12 20:42:19 +01002458def Test_did_emsg_reset()
2459 # executing an autocommand resets did_emsg, this should not result in a
2460 # builtin function considered failing
2461 var lines =<< trim END
2462 vim9script
2463 au BufWinLeave * #
2464 def Func()
Bram Moolenaar767034c2021-04-09 17:24:52 +02002465 popup_menu('', {callback: (a, b) => popup_create('', {})->popup_close()})
Bram Moolenaar57f799e2020-12-12 20:42:19 +01002466 eval [][0]
2467 enddef
2468 nno <F3> <cmd>call <sid>Func()<cr>
2469 feedkeys("\<F3>\e", 'xt')
2470 END
2471 writefile(lines, 'XemsgReset')
2472 assert_fails('so XemsgReset', ['E684:', 'E684:'], lines, 2)
2473 delete('XemsgReset')
2474 nunmap <F3>
2475 au! BufWinLeave
2476enddef
2477
Bram Moolenaar56602ba2020-12-05 21:22:08 +01002478def Test_abort_with_silent_call()
2479 var lines =<< trim END
2480 vim9script
2481 g:result = 'none'
2482 def Func()
2483 g:result += 3
2484 g:result = 'yes'
2485 enddef
2486 # error is silenced, but function aborts on error
2487 silent! Func()
2488 assert_equal('none', g:result)
2489 unlet g:result
2490 END
2491 CheckScriptSuccess(lines)
2492enddef
2493
Bram Moolenaarf665e972020-12-05 19:17:16 +01002494def Test_continues_with_silent_error()
2495 var lines =<< trim END
2496 vim9script
2497 g:result = 'none'
2498 def Func()
2499 silent! g:result += 3
2500 g:result = 'yes'
2501 enddef
2502 # error is silenced, function does not abort
2503 Func()
2504 assert_equal('yes', g:result)
2505 unlet g:result
2506 END
2507 CheckScriptSuccess(lines)
2508enddef
2509
Bram Moolenaaraf0df472020-12-02 20:51:22 +01002510def Test_abort_even_with_silent()
2511 var lines =<< trim END
2512 vim9script
2513 g:result = 'none'
2514 def Func()
2515 eval {-> ''}() .. '' .. {}['X']
2516 g:result = 'yes'
2517 enddef
Bram Moolenaarf665e972020-12-05 19:17:16 +01002518 silent! Func()
Bram Moolenaaraf0df472020-12-02 20:51:22 +01002519 assert_equal('none', g:result)
Bram Moolenaar4029cab2020-12-05 18:13:27 +01002520 unlet g:result
2521 END
2522 CheckScriptSuccess(lines)
2523enddef
2524
Bram Moolenaarf665e972020-12-05 19:17:16 +01002525def Test_cmdmod_silent_restored()
2526 var lines =<< trim END
2527 vim9script
2528 def Func()
2529 g:result = 'none'
2530 silent! g:result += 3
2531 g:result = 'none'
2532 g:result += 3
2533 enddef
2534 Func()
2535 END
2536 # can't use CheckScriptFailure, it ignores the :silent!
2537 var fname = 'Xdefsilent'
2538 writefile(lines, fname)
2539 var caught = 'no'
2540 try
2541 exe 'source ' .. fname
2542 catch /E1030:/
2543 caught = 'yes'
2544 assert_match('Func, line 4', v:throwpoint)
2545 endtry
2546 assert_equal('yes', caught)
2547 delete(fname)
2548enddef
2549
Bram Moolenaar2fecb532021-03-24 22:00:56 +01002550def Test_cmdmod_silent_nested()
2551 var lines =<< trim END
2552 vim9script
2553 var result = ''
2554
2555 def Error()
2556 result ..= 'Eb'
2557 eval [][0]
2558 result ..= 'Ea'
2559 enddef
2560
2561 def Crash()
2562 result ..= 'Cb'
2563 sil! Error()
2564 result ..= 'Ca'
2565 enddef
2566
2567 Crash()
2568 assert_equal('CbEbEaCa', result)
2569 END
2570 CheckScriptSuccess(lines)
2571enddef
2572
Bram Moolenaar4029cab2020-12-05 18:13:27 +01002573def Test_dict_member_with_silent()
2574 var lines =<< trim END
2575 vim9script
2576 g:result = 'none'
2577 var d: dict<any>
2578 def Func()
2579 try
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01002580 g:result = map([], (_, v) => ({}[v]))->join() .. d['']
Bram Moolenaar4029cab2020-12-05 18:13:27 +01002581 catch
2582 endtry
2583 enddef
2584 silent! Func()
2585 assert_equal('0', g:result)
2586 unlet g:result
Bram Moolenaaraf0df472020-12-02 20:51:22 +01002587 END
2588 CheckScriptSuccess(lines)
2589enddef
2590
Bram Moolenaarf9041332021-01-21 19:41:16 +01002591def Test_skip_cmds_with_silent()
2592 var lines =<< trim END
2593 vim9script
2594
2595 def Func(b: bool)
2596 Crash()
2597 enddef
2598
2599 def Crash()
2600 sil! :/not found/d _
2601 sil! :/not found/put _
2602 enddef
2603
2604 Func(true)
2605 END
2606 CheckScriptSuccess(lines)
2607enddef
2608
Bram Moolenaar5b3d1bb2020-12-22 12:20:08 +01002609def Test_opfunc()
2610 nnoremap <F3> <cmd>set opfunc=Opfunc<cr>g@
2611 def g:Opfunc(_: any): string
2612 setline(1, 'ASDF')
2613 return ''
2614 enddef
2615 new
2616 setline(1, 'asdf')
2617 feedkeys("\<F3>$", 'x')
2618 assert_equal('ASDF', getline(1))
2619
2620 bwipe!
2621 nunmap <F3>
2622enddef
2623
Bram Moolenaar077a4232020-12-22 18:33:27 +01002624" this was crashing on exit
2625def Test_nested_lambda_in_closure()
2626 var lines =<< trim END
2627 vim9script
Bram Moolenaar227c58a2021-04-28 20:40:44 +02002628 command WriteDone writefile(['Done'], 'XnestedDone')
Bram Moolenaar077a4232020-12-22 18:33:27 +01002629 def Outer()
2630 def g:Inner()
2631 echo map([1, 2, 3], {_, v -> v + 1})
2632 enddef
2633 g:Inner()
2634 enddef
2635 defcompile
Bram Moolenaar227c58a2021-04-28 20:40:44 +02002636 # not reached
Bram Moolenaar077a4232020-12-22 18:33:27 +01002637 END
Bram Moolenaar227c58a2021-04-28 20:40:44 +02002638 if !RunVim([], lines, '--clean -c WriteDone -c quit')
Bram Moolenaar077a4232020-12-22 18:33:27 +01002639 return
2640 endif
2641 assert_equal(['Done'], readfile('XnestedDone'))
2642 delete('XnestedDone')
2643enddef
2644
Bram Moolenaar04947cc2021-03-06 19:26:46 +01002645def Test_check_func_arg_types()
2646 var lines =<< trim END
2647 vim9script
2648 def F1(x: string): string
2649 return x
2650 enddef
2651
2652 def F2(x: number): number
2653 return x + 1
2654 enddef
2655
2656 def G(g: func): dict<func>
2657 return {f: g}
2658 enddef
2659
2660 def H(d: dict<func>): string
2661 return d.f('a')
2662 enddef
2663 END
2664
2665 CheckScriptSuccess(lines + ['echo H(G(F1))'])
2666 CheckScriptFailure(lines + ['echo H(G(F2))'], 'E1013:')
2667enddef
2668
Bram Moolenaar701cc6c2021-04-10 13:33:48 +02002669def Test_compile_error()
2670 var lines =<< trim END
2671 def g:Broken()
2672 echo 'a' + {}
2673 enddef
2674 call g:Broken()
2675 END
2676 # First call: compilation error
2677 CheckScriptFailure(lines, 'E1051: Wrong argument type for +')
2678
2679 # Second call won't try compiling again
2680 assert_fails('call g:Broken()', 'E1091: Function is not compiled: Broken')
Bram Moolenaar599410c2021-04-10 14:03:43 +02002681 delfunc g:Broken
2682
2683 # No error when compiling with :silent!
2684 lines =<< trim END
2685 def g:Broken()
2686 echo 'a' + []
2687 enddef
2688 silent! defcompile
2689 END
2690 CheckScriptSuccess(lines)
2691
2692 # Calling the function won't try compiling again
2693 assert_fails('call g:Broken()', 'E1091: Function is not compiled: Broken')
2694 delfunc g:Broken
Bram Moolenaar701cc6c2021-04-10 13:33:48 +02002695enddef
2696
Bram Moolenaar962c43b2021-04-10 17:18:09 +02002697def Test_ignored_argument()
2698 var lines =<< trim END
2699 vim9script
2700 def Ignore(_, _): string
2701 return 'yes'
2702 enddef
2703 assert_equal('yes', Ignore(1, 2))
2704
2705 func Ok(_)
2706 return a:_
2707 endfunc
2708 assert_equal('ok', Ok('ok'))
2709
2710 func Oktoo()
2711 let _ = 'too'
2712 return _
2713 endfunc
2714 assert_equal('too', Oktoo())
Bram Moolenaarda479c72021-04-10 21:01:38 +02002715
2716 assert_equal([[1], [2], [3]], range(3)->mapnew((_, v) => [v]->map((_, w) => w + 1)))
Bram Moolenaar962c43b2021-04-10 17:18:09 +02002717 END
2718 CheckScriptSuccess(lines)
2719
2720 lines =<< trim END
2721 def Ignore(_: string): string
2722 return _
2723 enddef
2724 defcompile
2725 END
2726 CheckScriptFailure(lines, 'E1181:', 1)
2727
2728 lines =<< trim END
2729 var _ = 1
2730 END
2731 CheckDefAndScriptFailure(lines, 'E1181:', 1)
Yegappan Lakshmanan34fcb692021-05-25 20:14:00 +02002732
2733 lines =<< trim END
2734 var x = _
2735 END
2736 CheckDefAndScriptFailure(lines, 'E1181:', 1)
Bram Moolenaar962c43b2021-04-10 17:18:09 +02002737enddef
2738
Bram Moolenaarbb8a7ce2021-04-10 20:10:26 +02002739def Test_too_many_arguments()
2740 var lines =<< trim END
2741 echo [0, 1, 2]->map(() => 123)
2742 END
2743 CheckDefExecAndScriptFailure(lines, 'E1106: 2 arguments too many', 1)
2744
2745 lines =<< trim END
2746 echo [0, 1, 2]->map((_) => 123)
2747 END
2748 CheckDefExecAndScriptFailure(lines, 'E1106: One argument too many', 1)
2749enddef
Bram Moolenaar077a4232020-12-22 18:33:27 +01002750
Bram Moolenaara6aa1642021-04-23 19:32:23 +02002751def Test_closing_brace_at_start_of_line()
2752 var lines =<< trim END
2753 def Func()
2754 enddef
2755 Func(
2756 )
2757 END
2758 call CheckDefAndScriptSuccess(lines)
2759enddef
2760
Bram Moolenaarf7779c62020-05-03 15:38:16 +02002761
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002762" vim: ts=8 sw=2 sts=2 expandtab tw=80 fdm=marker