blob: 6258b7d5d4687e34bc88d9abb7a14abdd410aa2c [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()
21endfunc
22
23def TestCompilingError()
Bram Moolenaarf4e8cdd2020-10-12 22:07:13 +020024 var lines =<< trim END
25 vim9script
26 def Fails()
27 echo nothing
28 enddef
29 defcompile
30 END
31 call writefile(lines, 'XTest_compile_error')
32 var buf = RunVimInTerminal('-S XTest_compile_error',
Bram Moolenaare0de1712020-12-02 17:36:54 +010033 {rows: 10, wait_for_ruler: 0})
Bram Moolenaar03dfde22021-02-14 13:17:22 +010034 call WaitForAssert(() => assert_match('Error detected while compiling command line.*Fails.*Variable not found: nothing',
35 Term_getlines(buf, range(1, 9))))
Bram Moolenaarf4e8cdd2020-10-12 22:07:13 +020036
37 # clean up
38 call StopVimInTerminal(buf)
39 call delete('XTest_compile_error')
40enddef
41
Bram Moolenaar0ba48e82020-11-17 18:23:19 +010042def CallRecursive(n: number): number
43 return CallRecursive(n + 1)
44enddef
45
46def CallMapRecursive(l: list<number>): number
Bram Moolenaar2949cfd2020-12-31 21:28:47 +010047 return map(l, (_, v) => CallMapRecursive([v]))[0]
Bram Moolenaar0ba48e82020-11-17 18:23:19 +010048enddef
49
50def Test_funcdepth_error()
51 set maxfuncdepth=10
52
53 var caught = false
54 try
55 CallRecursive(1)
56 catch /E132:/
57 caught = true
58 endtry
59 assert_true(caught)
60
61 caught = false
62 try
63 CallMapRecursive([1])
64 catch /E132:/
65 caught = true
66 endtry
67 assert_true(caught)
68
69 set maxfuncdepth&
70enddef
71
Bram Moolenaar5178b1b2021-01-01 18:43:51 +010072def Test_endfunc_enddef()
73 var lines =<< trim END
74 def Test()
75 echo 'test'
76 endfunc
77 enddef
78 END
79 CheckScriptFailure(lines, 'E1151:', 3)
80
81 lines =<< trim END
82 def Test()
83 func Nested()
84 echo 'test'
85 enddef
86 enddef
87 END
88 CheckScriptFailure(lines, 'E1152:', 4)
89enddef
90
Bram Moolenaarb8ba9b92021-01-01 18:54:34 +010091def Test_missing_endfunc_enddef()
92 var lines =<< trim END
93 vim9script
94 def Test()
95 echo 'test'
96 endef
97 END
98 CheckScriptFailure(lines, 'E1057:', 2)
99
100 lines =<< trim END
101 vim9script
102 func Some()
103 echo 'test'
104 enfffunc
105 END
106 CheckScriptFailure(lines, 'E126:', 2)
107enddef
108
Bram Moolenaar4efd9942021-01-24 21:14:20 +0100109def Test_white_space_before_paren()
110 var lines =<< trim END
111 vim9script
112 def Test ()
113 echo 'test'
114 enddef
115 END
116 CheckScriptFailure(lines, 'E1068:', 2)
117
118 lines =<< trim END
119 vim9script
120 func Test ()
121 echo 'test'
122 endfunc
123 END
124 CheckScriptFailure(lines, 'E1068:', 2)
125
126 lines =<< trim END
127 def Test ()
128 echo 'test'
129 enddef
130 END
131 CheckScriptFailure(lines, 'E1068:', 1)
132
133 lines =<< trim END
134 func Test ()
135 echo 'test'
136 endfunc
137 END
138 CheckScriptSuccess(lines)
139enddef
140
Bram Moolenaar832ea892021-01-08 21:55:26 +0100141def Test_enddef_dict_key()
142 var d = {
143 enddef: 'x',
144 endfunc: 'y',
145 }
146 assert_equal({enddef: 'x', endfunc: 'y'}, d)
147enddef
148
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200149def ReturnString(): string
150 return 'string'
151enddef
152
153def ReturnNumber(): number
154 return 123
155enddef
156
157let g:notNumber = 'string'
158
159def ReturnGlobal(): number
160 return g:notNumber
161enddef
162
163def Test_return_something()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +0200164 ReturnString()->assert_equal('string')
165 ReturnNumber()->assert_equal(123)
Bram Moolenaar5e654232020-09-16 15:22:00 +0200166 assert_fails('ReturnGlobal()', 'E1012: Type mismatch; expected number but got string', '', 1, 'ReturnGlobal')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200167enddef
168
Bram Moolenaare32e5162021-01-21 20:21:29 +0100169def Test_check_argument_type()
170 var lines =<< trim END
171 vim9script
172 def Val(a: number, b: number): number
173 return 0
174 enddef
175 def Func()
176 var x: any = true
177 Val(0, x)
178 enddef
179 disass Func
180 Func()
181 END
182 CheckScriptFailure(lines, 'E1013: Argument 2: type mismatch, expected number but got bool', 2)
183enddef
184
Bram Moolenaarefd88552020-06-18 20:50:10 +0200185def Test_missing_return()
186 CheckDefFailure(['def Missing(): number',
187 ' if g:cond',
188 ' echo "no return"',
189 ' else',
190 ' return 0',
191 ' endif'
192 'enddef'], 'E1027:')
193 CheckDefFailure(['def Missing(): number',
194 ' if g:cond',
195 ' return 1',
196 ' else',
197 ' echo "no return"',
198 ' endif'
199 'enddef'], 'E1027:')
200 CheckDefFailure(['def Missing(): number',
201 ' if g:cond',
202 ' return 1',
203 ' else',
204 ' return 2',
205 ' endif'
206 ' return 3'
207 'enddef'], 'E1095:')
208enddef
209
Bram Moolenaar403dc312020-10-17 19:29:51 +0200210def Test_return_bool()
211 var lines =<< trim END
212 vim9script
213 def MenuFilter(id: number, key: string): bool
214 return popup_filter_menu(id, key)
215 enddef
216 def YesnoFilter(id: number, key: string): bool
217 return popup_filter_yesno(id, key)
218 enddef
219 defcompile
220 END
221 CheckScriptSuccess(lines)
222enddef
223
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200224let s:nothing = 0
225def ReturnNothing()
226 s:nothing = 1
227 if true
228 return
229 endif
230 s:nothing = 2
231enddef
232
233def Test_return_nothing()
234 ReturnNothing()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +0200235 s:nothing->assert_equal(1)
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200236enddef
237
Bram Moolenaar648ea762021-01-15 19:04:32 +0100238def Test_return_invalid()
239 var lines =<< trim END
240 vim9script
241 def Func(): invalid
242 return xxx
243 enddef
244 defcompile
245 END
246 CheckScriptFailure(lines, 'E1010:', 2)
Bram Moolenaar31842cd2021-02-12 22:10:21 +0100247
248 lines =<< trim END
249 vim9script
250 def Test(Fun: func(number): number): list<number>
251 return map([1, 2, 3], (_, i) => Fun(i))
252 enddef
253 defcompile
254 def Inc(nr: number): nr
255 return nr + 2
256 enddef
257 echo Test(Inc)
258 END
259 # doing this twice was leaking memory
260 CheckScriptFailure(lines, 'E1010:')
261 CheckScriptFailure(lines, 'E1010:')
Bram Moolenaar648ea762021-01-15 19:04:32 +0100262enddef
263
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200264func Increment()
265 let g:counter += 1
266endfunc
267
268def Test_call_ufunc_count()
269 g:counter = 1
270 Increment()
271 Increment()
272 Increment()
Bram Moolenaarf5be8cd2020-07-17 20:36:00 +0200273 # works with and without :call
Bram Moolenaarc0c71e92020-09-11 19:09:48 +0200274 g:counter->assert_equal(4)
275 eval g:counter->assert_equal(4)
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200276 unlet g:counter
277enddef
278
279def MyVarargs(arg: string, ...rest: list<string>): string
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +0200280 var res = arg
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200281 for s in rest
282 res ..= ',' .. s
283 endfor
284 return res
285enddef
286
287def Test_call_varargs()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +0200288 MyVarargs('one')->assert_equal('one')
289 MyVarargs('one', 'two')->assert_equal('one,two')
290 MyVarargs('one', 'two', 'three')->assert_equal('one,two,three')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200291enddef
292
293def MyDefaultArgs(name = 'string'): string
294 return name
295enddef
296
Bram Moolenaare30f64b2020-07-15 19:48:20 +0200297def MyDefaultSecond(name: string, second: bool = true): string
298 return second ? name : 'none'
299enddef
300
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200301def Test_call_default_args()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +0200302 MyDefaultArgs()->assert_equal('string')
303 MyDefaultArgs('one')->assert_equal('one')
Bram Moolenaar9bd5d872020-09-06 21:47:48 +0200304 assert_fails('MyDefaultArgs("one", "two")', 'E118:', '', 3, 'Test_call_default_args')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200305
Bram Moolenaarc0c71e92020-09-11 19:09:48 +0200306 MyDefaultSecond('test')->assert_equal('test')
307 MyDefaultSecond('test', true)->assert_equal('test')
308 MyDefaultSecond('test', false)->assert_equal('none')
Bram Moolenaare30f64b2020-07-15 19:48:20 +0200309
Bram Moolenaar822ba242020-05-24 23:00:18 +0200310 CheckScriptFailure(['def Func(arg: number = asdf)', 'enddef', 'defcompile'], 'E1001:')
Bram Moolenaar2d870f82020-12-05 13:41:01 +0100311 delfunc g:Func
Bram Moolenaar77072282020-09-16 17:55:40 +0200312 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 +0100313 delfunc g:Func
Bram Moolenaar12bce952021-03-11 20:04:04 +0100314
315 var lines =<< trim END
316 vim9script
317 def Func(a = b == 0 ? 1 : 2, b = 0)
318 enddef
319 defcompile
320 END
321 CheckScriptFailure(lines, 'E1001: Variable not found: b')
Bram Moolenaar04b12692020-05-04 23:24:44 +0200322enddef
323
Bram Moolenaarcef12702021-01-04 14:09:43 +0100324def FuncWithComment( # comment
325 a: number, #comment
326 b: bool, # comment
327 c: string) #comment
328 assert_equal(4, a)
329 assert_equal(true, b)
330 assert_equal('yes', c)
331enddef
332
333def Test_func_with_comments()
334 FuncWithComment(4, true, 'yes')
335
336 var lines =<< trim END
337 def Func(# comment
338 arg: string)
339 enddef
340 END
341 CheckScriptFailure(lines, 'E125:', 1)
342
343 lines =<< trim END
344 def Func(
345 arg: string# comment
346 )
347 enddef
348 END
349 CheckScriptFailure(lines, 'E475:', 2)
350
351 lines =<< trim END
352 def Func(
353 arg: string
354 )# comment
355 enddef
356 END
357 CheckScriptFailure(lines, 'E488:', 3)
358enddef
359
Bram Moolenaar04b12692020-05-04 23:24:44 +0200360def Test_nested_function()
361 def Nested(arg: string): string
362 return 'nested ' .. arg
363 enddef
Bram Moolenaarc0c71e92020-09-11 19:09:48 +0200364 Nested('function')->assert_equal('nested function')
Bram Moolenaar04b12692020-05-04 23:24:44 +0200365
Bram Moolenaar0e65d3d2020-05-05 17:53:16 +0200366 CheckDefFailure(['def Nested()', 'enddef', 'Nested(66)'], 'E118:')
367 CheckDefFailure(['def Nested(arg: string)', 'enddef', 'Nested()'], 'E119:')
368
Bram Moolenaar04b12692020-05-04 23:24:44 +0200369 CheckDefFailure(['func Nested()', 'endfunc'], 'E1086:')
Bram Moolenaarbcbf4132020-08-01 22:35:13 +0200370 CheckDefFailure(['def s:Nested()', 'enddef'], 'E1075:')
371 CheckDefFailure(['def b:Nested()', 'enddef'], 'E1075:')
Bram Moolenaar8b848ca2020-09-10 22:28:01 +0200372
Bram Moolenaar54021752020-12-06 18:50:36 +0100373 var lines =<< trim END
374 def Outer()
375 def Inner()
376 # comment
377 enddef
378 def Inner()
379 enddef
380 enddef
381 END
382 CheckDefFailure(lines, 'E1073:')
383
384 lines =<< trim END
385 def Outer()
386 def Inner()
387 # comment
388 enddef
389 def! Inner()
390 enddef
391 enddef
392 END
393 CheckDefFailure(lines, 'E1117:')
394
395 # nested function inside conditional
396 # TODO: should it work when "thecount" is inside the "if"?
397 lines =<< trim END
398 vim9script
399 var thecount = 0
400 if true
401 def Test(): number
402 def TheFunc(): number
403 thecount += 1
404 return thecount
405 enddef
406 return TheFunc()
407 enddef
408 endif
409 defcompile
410 assert_equal(1, Test())
411 assert_equal(2, Test())
412 END
413 CheckScriptSuccess(lines)
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200414enddef
415
Bram Moolenaaradc8e442020-12-31 18:28:18 +0100416def Test_not_nested_function()
417 echo printf('%d',
418 function('len')('xxx'))
419enddef
420
Bram Moolenaaraf8edbb2020-08-01 00:03:09 +0200421func Test_call_default_args_from_func()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +0200422 call MyDefaultArgs()->assert_equal('string')
423 call MyDefaultArgs('one')->assert_equal('one')
Bram Moolenaar9bd5d872020-09-06 21:47:48 +0200424 call assert_fails('call MyDefaultArgs("one", "two")', 'E118:', '', 3, 'Test_call_default_args_from_func')
Bram Moolenaaraf8edbb2020-08-01 00:03:09 +0200425endfunc
426
Bram Moolenaar38ddf332020-07-31 22:05:04 +0200427def Test_nested_global_function()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +0200428 var lines =<< trim END
Bram Moolenaar38ddf332020-07-31 22:05:04 +0200429 vim9script
430 def Outer()
431 def g:Inner(): string
432 return 'inner'
433 enddef
434 enddef
Bram Moolenaaraf8edbb2020-08-01 00:03:09 +0200435 defcompile
436 Outer()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +0200437 g:Inner()->assert_equal('inner')
Bram Moolenaaraf8edbb2020-08-01 00:03:09 +0200438 delfunc g:Inner
439 Outer()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +0200440 g:Inner()->assert_equal('inner')
Bram Moolenaaraf8edbb2020-08-01 00:03:09 +0200441 delfunc g:Inner
442 Outer()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +0200443 g:Inner()->assert_equal('inner')
Bram Moolenaaraf8edbb2020-08-01 00:03:09 +0200444 delfunc g:Inner
Bram Moolenaar38ddf332020-07-31 22:05:04 +0200445 END
446 CheckScriptSuccess(lines)
Bram Moolenaar2c79e9d2020-08-01 18:57:52 +0200447
448 lines =<< trim END
449 vim9script
450 def Outer()
451 def g:Inner(): string
452 return 'inner'
453 enddef
454 enddef
455 defcompile
456 Outer()
457 Outer()
458 END
459 CheckScriptFailure(lines, "E122:")
Bram Moolenaarcd45ed02020-12-22 17:35:54 +0100460 delfunc g:Inner
Bram Moolenaarad486a02020-08-01 23:22:18 +0200461
462 lines =<< trim END
463 vim9script
Bram Moolenaar58a52f22020-12-22 18:56:55 +0100464 def Outer()
465 def g:Inner()
Bram Moolenaar2949cfd2020-12-31 21:28:47 +0100466 echo map([1, 2, 3], (_, v) => v + 1)
Bram Moolenaar58a52f22020-12-22 18:56:55 +0100467 enddef
468 g:Inner()
469 enddef
470 Outer()
471 END
472 CheckScriptSuccess(lines)
473 delfunc g:Inner
474
475 lines =<< trim END
476 vim9script
Bram Moolenaarad486a02020-08-01 23:22:18 +0200477 def Func()
478 echo 'script'
479 enddef
480 def Outer()
481 def Func()
482 echo 'inner'
483 enddef
484 enddef
485 defcompile
486 END
487 CheckScriptFailure(lines, "E1073:")
Bram Moolenaar38ddf332020-07-31 22:05:04 +0200488enddef
489
Bram Moolenaar6abdcf82020-11-22 18:15:44 +0100490def DefListAll()
491 def
492enddef
493
494def DefListOne()
495 def DefListOne
496enddef
497
498def DefListMatches()
499 def /DefList
500enddef
501
502def Test_nested_def_list()
503 var funcs = split(execute('call DefListAll()'), "\n")
504 assert_true(len(funcs) > 10)
505 assert_true(funcs->index('def DefListAll()') >= 0)
506
507 funcs = split(execute('call DefListOne()'), "\n")
508 assert_equal([' def DefListOne()', '1 def DefListOne', ' enddef'], funcs)
509
510 funcs = split(execute('call DefListMatches()'), "\n")
511 assert_true(len(funcs) >= 3)
512 assert_true(funcs->index('def DefListAll()') >= 0)
513 assert_true(funcs->index('def DefListOne()') >= 0)
514 assert_true(funcs->index('def DefListMatches()') >= 0)
Bram Moolenaar54021752020-12-06 18:50:36 +0100515
516 var lines =<< trim END
517 vim9script
518 def Func()
519 def +Func+
520 enddef
521 defcompile
522 END
523 CheckScriptFailure(lines, 'E476:', 1)
Bram Moolenaar6abdcf82020-11-22 18:15:44 +0100524enddef
525
Bram Moolenaar333894b2020-08-01 18:53:07 +0200526def Test_global_local_function()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +0200527 var lines =<< trim END
Bram Moolenaar333894b2020-08-01 18:53:07 +0200528 vim9script
529 def g:Func(): string
530 return 'global'
531 enddef
532 def Func(): string
533 return 'local'
534 enddef
Bram Moolenaarc0c71e92020-09-11 19:09:48 +0200535 g:Func()->assert_equal('global')
536 Func()->assert_equal('local')
Bram Moolenaar2d870f82020-12-05 13:41:01 +0100537 delfunc g:Func
Bram Moolenaar333894b2020-08-01 18:53:07 +0200538 END
539 CheckScriptSuccess(lines)
Bram Moolenaar035d6e92020-08-11 22:30:42 +0200540
541 lines =<< trim END
542 vim9script
543 def g:Funcy()
544 echo 'funcy'
545 enddef
546 s:Funcy()
547 END
548 CheckScriptFailure(lines, 'E117:')
Bram Moolenaar333894b2020-08-01 18:53:07 +0200549enddef
550
Bram Moolenaar0f769812020-09-12 18:32:34 +0200551def Test_local_function_shadows_global()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +0200552 var lines =<< trim END
Bram Moolenaar0f769812020-09-12 18:32:34 +0200553 vim9script
554 def g:Gfunc(): string
555 return 'global'
556 enddef
557 def AnotherFunc(): number
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +0200558 var Gfunc = function('len')
Bram Moolenaar0f769812020-09-12 18:32:34 +0200559 return Gfunc('testing')
560 enddef
561 g:Gfunc()->assert_equal('global')
562 AnotherFunc()->assert_equal(7)
563 delfunc g:Gfunc
564 END
565 CheckScriptSuccess(lines)
566
567 lines =<< trim END
568 vim9script
569 def g:Func(): string
570 return 'global'
571 enddef
572 def AnotherFunc()
573 g:Func = function('len')
574 enddef
575 AnotherFunc()
576 END
577 CheckScriptFailure(lines, 'E705:')
578 delfunc g:Func
579enddef
580
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200581func TakesOneArg(arg)
582 echo a:arg
583endfunc
584
585def Test_call_wrong_args()
Bram Moolenaard2c61702020-09-06 15:58:36 +0200586 CheckDefFailure(['TakesOneArg()'], 'E119:')
587 CheckDefFailure(['TakesOneArg(11, 22)'], 'E118:')
588 CheckDefFailure(['bufnr(xxx)'], 'E1001:')
589 CheckScriptFailure(['def Func(Ref: func(s: string))'], 'E475:')
Bram Moolenaaree8580e2020-08-28 17:19:07 +0200590
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +0200591 var lines =<< trim END
Bram Moolenaaree8580e2020-08-28 17:19:07 +0200592 vim9script
593 def Func(s: string)
594 echo s
595 enddef
596 Func([])
597 END
Bram Moolenaar77072282020-09-16 17:55:40 +0200598 CheckScriptFailure(lines, 'E1013: Argument 1: type mismatch, expected string but got list<unknown>', 5)
Bram Moolenaarb185a402020-09-18 22:42:00 +0200599
600 lines =<< trim END
601 vim9script
Bram Moolenaarb4893b82021-02-21 22:20:24 +0100602 var name = 'piet'
603 def FuncOne(name: string)
604 echo nr
605 enddef
606 END
Bram Moolenaar057e84a2021-02-28 16:55:11 +0100607 CheckScriptFailure(lines, 'E1168:')
Bram Moolenaarb4893b82021-02-21 22:20:24 +0100608
609 lines =<< trim END
610 vim9script
Bram Moolenaarb185a402020-09-18 22:42:00 +0200611 def FuncOne(nr: number)
612 echo nr
613 enddef
614 def FuncTwo()
615 FuncOne()
616 enddef
617 defcompile
618 END
619 writefile(lines, 'Xscript')
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +0200620 var didCatch = false
Bram Moolenaarb185a402020-09-18 22:42:00 +0200621 try
622 source Xscript
623 catch
624 assert_match('E119: Not enough arguments for function: <SNR>\d\+_FuncOne', v:exception)
625 assert_match('Xscript\[8\]..function <SNR>\d\+_FuncTwo, line 1', v:throwpoint)
626 didCatch = true
627 endtry
628 assert_true(didCatch)
629
630 lines =<< trim END
631 vim9script
632 def FuncOne(nr: number)
633 echo nr
634 enddef
635 def FuncTwo()
636 FuncOne(1, 2)
637 enddef
638 defcompile
639 END
640 writefile(lines, 'Xscript')
641 didCatch = false
642 try
643 source Xscript
644 catch
645 assert_match('E118: Too many arguments for function: <SNR>\d\+_FuncOne', v:exception)
646 assert_match('Xscript\[8\]..function <SNR>\d\+_FuncTwo, line 1', v:throwpoint)
647 didCatch = true
648 endtry
649 assert_true(didCatch)
650
651 delete('Xscript')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200652enddef
653
Bram Moolenaar50824712020-12-20 21:10:17 +0100654def Test_call_funcref_wrong_args()
655 var head =<< trim END
656 vim9script
657 def Func3(a1: string, a2: number, a3: list<number>)
658 echo a1 .. a2 .. a3[0]
659 enddef
660 def Testme()
661 var funcMap: dict<func> = {func: Func3}
662 END
663 var tail =<< trim END
664 enddef
665 Testme()
666 END
667 CheckScriptSuccess(head + ["funcMap['func']('str', 123, [1, 2, 3])"] + tail)
668
669 CheckScriptFailure(head + ["funcMap['func']('str', 123)"] + tail, 'E119:')
670 CheckScriptFailure(head + ["funcMap['func']('str', 123, [1], 4)"] + tail, 'E118:')
Bram Moolenaar32b3f822021-01-06 21:59:39 +0100671
672 var lines =<< trim END
673 vim9script
674 var Ref: func(number): any
675 Ref = (j) => !j
676 echo Ref(false)
677 END
678 CheckScriptFailure(lines, 'E1013: Argument 1: type mismatch, expected number but got bool', 4)
679
680 lines =<< trim END
681 vim9script
682 var Ref: func(number): any
683 Ref = (j) => !j
684 call Ref(false)
685 END
686 CheckScriptFailure(lines, 'E1013: Argument 1: type mismatch, expected number but got bool', 4)
Bram Moolenaar50824712020-12-20 21:10:17 +0100687enddef
688
Bram Moolenaarb4d16cb2020-11-05 18:45:46 +0100689def Test_call_lambda_args()
Bram Moolenaar2949cfd2020-12-31 21:28:47 +0100690 CheckDefFailure(['echo ((i) => 0)()'],
691 'E119: Not enough arguments for function: ((i) => 0)()')
Bram Moolenaarb4d16cb2020-11-05 18:45:46 +0100692
693 var lines =<< trim END
Bram Moolenaar2949cfd2020-12-31 21:28:47 +0100694 var Ref = (x: number, y: number) => x + y
Bram Moolenaarb4d16cb2020-11-05 18:45:46 +0100695 echo Ref(1, 'x')
696 END
697 CheckDefFailure(lines, 'E1013: Argument 2: type mismatch, expected number but got string')
Bram Moolenaare68b02a2021-01-03 13:09:51 +0100698
699 lines =<< trim END
700 var Ref: func(job, string, number)
701 Ref = (x, y) => 0
702 END
703 CheckDefAndScriptFailure(lines, 'E1012:')
704
705 lines =<< trim END
706 var Ref: func(job, string)
707 Ref = (x, y, z) => 0
708 END
709 CheckDefAndScriptFailure(lines, 'E1012:')
Bram Moolenaar057e84a2021-02-28 16:55:11 +0100710
711 lines =<< trim END
712 var one = 1
713 var l = [1, 2, 3]
714 echo map(l, (one) => one)
715 END
716 CheckDefFailure(lines, 'E1167:')
717 CheckScriptFailure(['vim9script'] + lines, 'E1168:')
718
719 lines =<< trim END
720 def ShadowLocal()
721 var one = 1
722 var l = [1, 2, 3]
723 echo map(l, (one) => one)
724 enddef
725 END
726 CheckDefFailure(lines, 'E1167:')
727
728 lines =<< trim END
729 def Shadowarg(one: number)
730 var l = [1, 2, 3]
731 echo map(l, (one) => one)
732 enddef
733 END
734 CheckDefFailure(lines, 'E1167:')
Bram Moolenaarb4d16cb2020-11-05 18:45:46 +0100735enddef
736
Bram Moolenaar0346b792021-01-31 22:18:29 +0100737def Test_lambda_return_type()
738 var lines =<< trim END
739 var Ref = (): => 123
740 END
741 CheckDefAndScriptFailure(lines, 'E1157:', 1)
742enddef
743
Bram Moolenaar709664c2020-12-12 14:33:41 +0100744def Test_lambda_uses_assigned_var()
745 CheckDefSuccess([
746 'var x: any = "aaa"'
Bram Moolenaar2949cfd2020-12-31 21:28:47 +0100747 'x = filter(["bbb"], (_, v) => v =~ x)'])
Bram Moolenaar709664c2020-12-12 14:33:41 +0100748enddef
749
Bram Moolenaar18062fc2021-03-05 21:35:47 +0100750def Test_pass_legacy_lambda_to_def_func()
751 var lines =<< trim END
752 vim9script
753 func Foo()
754 eval s:Bar({x -> 0})
755 endfunc
756 def Bar(y: any)
757 enddef
758 Foo()
759 END
760 CheckScriptSuccess(lines)
761enddef
762
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200763" Default arg and varargs
764def MyDefVarargs(one: string, two = 'foo', ...rest: list<string>): string
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +0200765 var res = one .. ',' .. two
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200766 for s in rest
767 res ..= ',' .. s
768 endfor
769 return res
770enddef
771
772def Test_call_def_varargs()
Bram Moolenaar9bd5d872020-09-06 21:47:48 +0200773 assert_fails('MyDefVarargs()', 'E119:', '', 1, 'Test_call_def_varargs')
Bram Moolenaarc0c71e92020-09-11 19:09:48 +0200774 MyDefVarargs('one')->assert_equal('one,foo')
775 MyDefVarargs('one', 'two')->assert_equal('one,two')
776 MyDefVarargs('one', 'two', 'three')->assert_equal('one,two,three')
Bram Moolenaar24aa48b2020-07-25 16:33:02 +0200777 CheckDefFailure(['MyDefVarargs("one", 22)'],
Bram Moolenaar77072282020-09-16 17:55:40 +0200778 'E1013: Argument 2: type mismatch, expected string but got number')
Bram Moolenaar24aa48b2020-07-25 16:33:02 +0200779 CheckDefFailure(['MyDefVarargs("one", "two", 123)'],
Bram Moolenaar77072282020-09-16 17:55:40 +0200780 'E1013: Argument 3: type mismatch, expected string but got number')
Bram Moolenaar24aa48b2020-07-25 16:33:02 +0200781
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +0200782 var lines =<< trim END
Bram Moolenaar24aa48b2020-07-25 16:33:02 +0200783 vim9script
784 def Func(...l: list<string>)
785 echo l
786 enddef
787 Func('a', 'b', 'c')
788 END
789 CheckScriptSuccess(lines)
790
791 lines =<< trim END
792 vim9script
793 def Func(...l: list<string>)
794 echo l
795 enddef
796 Func()
797 END
798 CheckScriptSuccess(lines)
799
800 lines =<< trim END
801 vim9script
Bram Moolenaar2f8cbc42020-09-16 17:22:59 +0200802 def Func(...l: any)
803 echo l
804 enddef
805 Func(0)
806 END
807 CheckScriptSuccess(lines)
808
809 lines =<< trim END
810 vim9script
Bram Moolenaar28022722020-09-21 22:02:49 +0200811 def Func(..._l: list<string>)
812 echo _l
813 enddef
814 Func('a', 'b', 'c')
815 END
816 CheckScriptSuccess(lines)
817
818 lines =<< trim END
819 vim9script
Bram Moolenaar24aa48b2020-07-25 16:33:02 +0200820 def Func(...l: list<string>)
821 echo l
822 enddef
823 Func(1, 2, 3)
824 END
Bram Moolenaar77072282020-09-16 17:55:40 +0200825 CheckScriptFailure(lines, 'E1013: Argument 1: type mismatch')
Bram Moolenaar24aa48b2020-07-25 16:33:02 +0200826
827 lines =<< trim END
828 vim9script
829 def Func(...l: list<string>)
830 echo l
831 enddef
832 Func('a', 9)
833 END
Bram Moolenaar77072282020-09-16 17:55:40 +0200834 CheckScriptFailure(lines, 'E1013: Argument 2: type mismatch')
Bram Moolenaar24aa48b2020-07-25 16:33:02 +0200835
836 lines =<< trim END
837 vim9script
838 def Func(...l: list<string>)
839 echo l
840 enddef
841 Func(1, 'a')
842 END
Bram Moolenaar77072282020-09-16 17:55:40 +0200843 CheckScriptFailure(lines, 'E1013: Argument 1: type mismatch')
Bram Moolenaar4f53b792021-02-07 15:59:49 +0100844
845 lines =<< trim END
846 vim9script
847 def Func( # some comment
848 ...l = []
849 )
850 echo l
851 enddef
852 END
853 CheckScriptFailure(lines, 'E1160:')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200854enddef
855
Bram Moolenaar1378fbc2020-04-11 20:50:33 +0200856let s:value = ''
857
858def FuncOneDefArg(opt = 'text')
859 s:value = opt
860enddef
861
862def FuncTwoDefArg(nr = 123, opt = 'text'): string
863 return nr .. opt
864enddef
865
866def FuncVarargs(...arg: list<string>): string
867 return join(arg, ',')
868enddef
869
870def Test_func_type_varargs()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +0200871 var RefDefArg: func(?string)
Bram Moolenaar1378fbc2020-04-11 20:50:33 +0200872 RefDefArg = FuncOneDefArg
873 RefDefArg()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +0200874 s:value->assert_equal('text')
Bram Moolenaar1378fbc2020-04-11 20:50:33 +0200875 RefDefArg('some')
Bram Moolenaarc0c71e92020-09-11 19:09:48 +0200876 s:value->assert_equal('some')
Bram Moolenaar1378fbc2020-04-11 20:50:33 +0200877
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +0200878 var RefDef2Arg: func(?number, ?string): string
Bram Moolenaar1378fbc2020-04-11 20:50:33 +0200879 RefDef2Arg = FuncTwoDefArg
Bram Moolenaarc0c71e92020-09-11 19:09:48 +0200880 RefDef2Arg()->assert_equal('123text')
881 RefDef2Arg(99)->assert_equal('99text')
882 RefDef2Arg(77, 'some')->assert_equal('77some')
Bram Moolenaar1378fbc2020-04-11 20:50:33 +0200883
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +0200884 CheckDefFailure(['var RefWrong: func(string?)'], 'E1010:')
885 CheckDefFailure(['var RefWrong: func(?string, string)'], 'E1007:')
Bram Moolenaar1378fbc2020-04-11 20:50:33 +0200886
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +0200887 var RefVarargs: func(...list<string>): string
Bram Moolenaar1378fbc2020-04-11 20:50:33 +0200888 RefVarargs = FuncVarargs
Bram Moolenaarc0c71e92020-09-11 19:09:48 +0200889 RefVarargs()->assert_equal('')
890 RefVarargs('one')->assert_equal('one')
891 RefVarargs('one', 'two')->assert_equal('one,two')
Bram Moolenaar1378fbc2020-04-11 20:50:33 +0200892
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +0200893 CheckDefFailure(['var RefWrong: func(...list<string>, string)'], 'E110:')
894 CheckDefFailure(['var RefWrong: func(...list<string>, ?string)'], 'E110:')
Bram Moolenaar1378fbc2020-04-11 20:50:33 +0200895enddef
896
Bram Moolenaar0b76b422020-04-07 22:05:08 +0200897" Only varargs
898def MyVarargsOnly(...args: list<string>): string
899 return join(args, ',')
900enddef
901
902def Test_call_varargs_only()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +0200903 MyVarargsOnly()->assert_equal('')
904 MyVarargsOnly('one')->assert_equal('one')
905 MyVarargsOnly('one', 'two')->assert_equal('one,two')
Bram Moolenaar77072282020-09-16 17:55:40 +0200906 CheckDefFailure(['MyVarargsOnly(1)'], 'E1013: Argument 1: type mismatch, expected string but got number')
907 CheckDefFailure(['MyVarargsOnly("one", 2)'], 'E1013: Argument 2: type mismatch, expected string but got number')
Bram Moolenaar0b76b422020-04-07 22:05:08 +0200908enddef
909
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200910def Test_using_var_as_arg()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +0200911 writefile(['def Func(x: number)', 'var x = 234', 'enddef', 'defcompile'], 'Xdef')
Bram Moolenaar9bd5d872020-09-06 21:47:48 +0200912 assert_fails('so Xdef', 'E1006:', '', 1, 'Func')
Bram Moolenaard2c61702020-09-06 15:58:36 +0200913 delete('Xdef')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200914enddef
915
Bram Moolenaarcb2bdb12020-05-10 22:53:56 +0200916def DictArg(arg: dict<string>)
917 arg['key'] = 'value'
918enddef
919
920def ListArg(arg: list<string>)
921 arg[0] = 'value'
922enddef
923
924def Test_assign_to_argument()
Bram Moolenaarf5be8cd2020-07-17 20:36:00 +0200925 # works for dict and list
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +0200926 var d: dict<string> = {}
Bram Moolenaarcb2bdb12020-05-10 22:53:56 +0200927 DictArg(d)
Bram Moolenaarc0c71e92020-09-11 19:09:48 +0200928 d['key']->assert_equal('value')
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +0200929 var l: list<string> = []
Bram Moolenaarcb2bdb12020-05-10 22:53:56 +0200930 ListArg(l)
Bram Moolenaarc0c71e92020-09-11 19:09:48 +0200931 l[0]->assert_equal('value')
Bram Moolenaarcb2bdb12020-05-10 22:53:56 +0200932
Bram Moolenaard2c61702020-09-06 15:58:36 +0200933 CheckScriptFailure(['def Func(arg: number)', 'arg = 3', 'enddef', 'defcompile'], 'E1090:')
Bram Moolenaar2d870f82020-12-05 13:41:01 +0100934 delfunc! g:Func
Bram Moolenaarcb2bdb12020-05-10 22:53:56 +0200935enddef
936
Bram Moolenaarb816dae2020-09-20 22:04:00 +0200937" These argument names are reserved in legacy functions.
938def WithReservedNames(firstline: string, lastline: string): string
939 return firstline .. lastline
940enddef
941
942def Test_argument_names()
943 assert_equal('OK', WithReservedNames('O', 'K'))
944enddef
945
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200946def Test_call_func_defined_later()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +0200947 g:DefinedLater('one')->assert_equal('one')
Bram Moolenaar9bd5d872020-09-06 21:47:48 +0200948 assert_fails('NotDefined("one")', 'E117:', '', 2, 'Test_call_func_defined_later')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200949enddef
950
Bram Moolenaar1df8b3f2020-04-23 18:13:23 +0200951func DefinedLater(arg)
952 return a:arg
953endfunc
954
955def Test_call_funcref()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +0200956 g:SomeFunc('abc')->assert_equal(3)
Bram Moolenaar9bd5d872020-09-06 21:47:48 +0200957 assert_fails('NotAFunc()', 'E117:', '', 2, 'Test_call_funcref') # comment after call
958 assert_fails('g:NotAFunc()', 'E117:', '', 3, 'Test_call_funcref')
Bram Moolenaar2f1980f2020-07-22 19:30:06 +0200959
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +0200960 var lines =<< trim END
Bram Moolenaar2f1980f2020-07-22 19:30:06 +0200961 vim9script
962 def RetNumber(): number
963 return 123
964 enddef
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +0200965 var Funcref: func: number = function('RetNumber')
Bram Moolenaarc0c71e92020-09-11 19:09:48 +0200966 Funcref()->assert_equal(123)
Bram Moolenaar2f1980f2020-07-22 19:30:06 +0200967 END
968 CheckScriptSuccess(lines)
Bram Moolenaar0f60e802020-07-22 20:16:11 +0200969
970 lines =<< trim END
971 vim9script
972 def RetNumber(): number
973 return 123
974 enddef
975 def Bar(F: func: number): number
976 return F()
977 enddef
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +0200978 var Funcref = function('RetNumber')
Bram Moolenaarc0c71e92020-09-11 19:09:48 +0200979 Bar(Funcref)->assert_equal(123)
Bram Moolenaar0f60e802020-07-22 20:16:11 +0200980 END
981 CheckScriptSuccess(lines)
Bram Moolenaarbfba8652020-07-23 20:09:10 +0200982
983 lines =<< trim END
984 vim9script
985 def UseNumber(nr: number)
986 echo nr
987 enddef
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +0200988 var Funcref: func(number) = function('UseNumber')
Bram Moolenaarbfba8652020-07-23 20:09:10 +0200989 Funcref(123)
990 END
991 CheckScriptSuccess(lines)
Bram Moolenaarb8070e32020-07-23 20:56:04 +0200992
993 lines =<< trim END
994 vim9script
995 def UseNumber(nr: number)
996 echo nr
997 enddef
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +0200998 var Funcref: func(string) = function('UseNumber')
Bram Moolenaarb8070e32020-07-23 20:56:04 +0200999 END
Bram Moolenaar5e654232020-09-16 15:22:00 +02001000 CheckScriptFailure(lines, 'E1012: Type mismatch; expected func(string) but got func(number)')
Bram Moolenaar4fc224c2020-07-26 17:56:25 +02001001
1002 lines =<< trim END
1003 vim9script
1004 def EchoNr(nr = 34)
1005 g:echo = nr
1006 enddef
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001007 var Funcref: func(?number) = function('EchoNr')
Bram Moolenaar4fc224c2020-07-26 17:56:25 +02001008 Funcref()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001009 g:echo->assert_equal(34)
Bram Moolenaar4fc224c2020-07-26 17:56:25 +02001010 Funcref(123)
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001011 g:echo->assert_equal(123)
Bram Moolenaar4fc224c2020-07-26 17:56:25 +02001012 END
1013 CheckScriptSuccess(lines)
Bram Moolenaarace61322020-07-26 18:16:58 +02001014
1015 lines =<< trim END
1016 vim9script
1017 def EchoList(...l: list<number>)
1018 g:echo = l
1019 enddef
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001020 var Funcref: func(...list<number>) = function('EchoList')
Bram Moolenaarace61322020-07-26 18:16:58 +02001021 Funcref()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001022 g:echo->assert_equal([])
Bram Moolenaarace61322020-07-26 18:16:58 +02001023 Funcref(1, 2, 3)
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001024 g:echo->assert_equal([1, 2, 3])
Bram Moolenaarace61322020-07-26 18:16:58 +02001025 END
1026 CheckScriptSuccess(lines)
Bram Moolenaar01865ad2020-07-26 18:33:09 +02001027
1028 lines =<< trim END
1029 vim9script
1030 def OptAndVar(nr: number, opt = 12, ...l: list<number>): number
1031 g:optarg = opt
1032 g:listarg = l
1033 return nr
1034 enddef
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001035 var Funcref: func(number, ?number, ...list<number>): number = function('OptAndVar')
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001036 Funcref(10)->assert_equal(10)
1037 g:optarg->assert_equal(12)
1038 g:listarg->assert_equal([])
Bram Moolenaar01865ad2020-07-26 18:33:09 +02001039
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001040 Funcref(11, 22)->assert_equal(11)
1041 g:optarg->assert_equal(22)
1042 g:listarg->assert_equal([])
Bram Moolenaar01865ad2020-07-26 18:33:09 +02001043
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001044 Funcref(17, 18, 1, 2, 3)->assert_equal(17)
1045 g:optarg->assert_equal(18)
1046 g:listarg->assert_equal([1, 2, 3])
Bram Moolenaar01865ad2020-07-26 18:33:09 +02001047 END
1048 CheckScriptSuccess(lines)
Bram Moolenaar1df8b3f2020-04-23 18:13:23 +02001049enddef
1050
1051let SomeFunc = function('len')
1052let NotAFunc = 'text'
1053
Bram Moolenaar99aaf0c2020-04-12 14:39:53 +02001054def CombineFuncrefTypes()
Bram Moolenaarf5be8cd2020-07-17 20:36:00 +02001055 # same arguments, different return type
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001056 var Ref1: func(bool): string
1057 var Ref2: func(bool): number
1058 var Ref3: func(bool): any
Bram Moolenaar99aaf0c2020-04-12 14:39:53 +02001059 Ref3 = g:cond ? Ref1 : Ref2
1060
Bram Moolenaarf5be8cd2020-07-17 20:36:00 +02001061 # different number of arguments
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001062 var Refa1: func(bool): number
1063 var Refa2: func(bool, number): number
1064 var Refa3: func: number
Bram Moolenaar99aaf0c2020-04-12 14:39:53 +02001065 Refa3 = g:cond ? Refa1 : Refa2
1066
Bram Moolenaarf5be8cd2020-07-17 20:36:00 +02001067 # different argument types
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001068 var Refb1: func(bool, string): number
1069 var Refb2: func(string, number): number
1070 var Refb3: func(any, any): number
Bram Moolenaar99aaf0c2020-04-12 14:39:53 +02001071 Refb3 = g:cond ? Refb1 : Refb2
1072enddef
1073
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001074def FuncWithForwardCall()
Bram Moolenaar1df8b3f2020-04-23 18:13:23 +02001075 return g:DefinedEvenLater("yes")
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001076enddef
1077
1078def DefinedEvenLater(arg: string): string
1079 return arg
1080enddef
1081
1082def Test_error_in_nested_function()
Bram Moolenaarf5be8cd2020-07-17 20:36:00 +02001083 # Error in called function requires unwinding the call stack.
Bram Moolenaar44d66522020-09-06 22:26:57 +02001084 assert_fails('FuncWithForwardCall()', 'E1096:', '', 1, 'FuncWithForwardCall')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001085enddef
1086
1087def Test_return_type_wrong()
Bram Moolenaar5a849da2020-08-08 16:47:30 +02001088 CheckScriptFailure([
1089 'def Func(): number',
1090 'return "a"',
1091 'enddef',
1092 'defcompile'], 'expected number but got string')
Bram Moolenaar2d870f82020-12-05 13:41:01 +01001093 delfunc! g:Func
Bram Moolenaar5a849da2020-08-08 16:47:30 +02001094 CheckScriptFailure([
1095 'def Func(): string',
1096 'return 1',
1097 'enddef',
1098 'defcompile'], 'expected string but got number')
Bram Moolenaar2d870f82020-12-05 13:41:01 +01001099 delfunc! g:Func
Bram Moolenaar5a849da2020-08-08 16:47:30 +02001100 CheckScriptFailure([
1101 'def Func(): void',
1102 'return "a"',
1103 'enddef',
1104 'defcompile'],
1105 'E1096: Returning a value in a function without a return type')
Bram Moolenaar2d870f82020-12-05 13:41:01 +01001106 delfunc! g:Func
Bram Moolenaar5a849da2020-08-08 16:47:30 +02001107 CheckScriptFailure([
1108 'def Func()',
1109 'return "a"',
1110 'enddef',
1111 'defcompile'],
1112 'E1096: Returning a value in a function without a return type')
Bram Moolenaar2d870f82020-12-05 13:41:01 +01001113 delfunc! g:Func
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001114
Bram Moolenaar5a849da2020-08-08 16:47:30 +02001115 CheckScriptFailure([
1116 'def Func(): number',
1117 'return',
1118 'enddef',
1119 'defcompile'], 'E1003:')
Bram Moolenaar2d870f82020-12-05 13:41:01 +01001120 delfunc! g:Func
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001121
1122 CheckScriptFailure(['def Func(): list', 'return []', 'enddef'], 'E1008:')
Bram Moolenaar2d870f82020-12-05 13:41:01 +01001123 delfunc! g:Func
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001124 CheckScriptFailure(['def Func(): dict', 'return {}', 'enddef'], 'E1008:')
Bram Moolenaar2d870f82020-12-05 13:41:01 +01001125 delfunc! g:Func
Bram Moolenaaree4e0c12020-04-06 21:35:05 +02001126 CheckScriptFailure(['def Func()', 'return 1'], 'E1057:')
Bram Moolenaar2d870f82020-12-05 13:41:01 +01001127 delfunc! g:Func
Bram Moolenaar5a849da2020-08-08 16:47:30 +02001128
1129 CheckScriptFailure([
1130 'vim9script',
1131 'def FuncB()',
1132 ' return 123',
1133 'enddef',
1134 'def FuncA()',
1135 ' FuncB()',
1136 'enddef',
1137 'defcompile'], 'E1096:')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001138enddef
1139
1140def Test_arg_type_wrong()
1141 CheckScriptFailure(['def Func3(items: list)', 'echo "a"', 'enddef'], 'E1008: Missing <type>')
Bram Moolenaaree4e0c12020-04-06 21:35:05 +02001142 CheckScriptFailure(['def Func4(...)', 'echo "a"', 'enddef'], 'E1055: Missing name after ...')
Bram Moolenaarf93c7fe2020-04-23 22:16:53 +02001143 CheckScriptFailure(['def Func5(items:string)', 'echo "a"'], 'E1069:')
Bram Moolenaar6e949782020-04-13 17:21:00 +02001144 CheckScriptFailure(['def Func5(items)', 'echo "a"'], 'E1077:')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001145enddef
1146
Bram Moolenaar608d78f2021-03-06 22:33:12 +01001147def Test_white_space_after_comma()
1148 var lines =<< trim END
1149 vim9script
1150 def Func(a: number,b: number)
1151 enddef
1152 END
1153 CheckScriptFailure(lines, 'E1069:')
1154
1155 # OK in legacy function
1156 lines =<< trim END
1157 vim9script
1158 func Func(a,b)
1159 endfunc
1160 END
1161 CheckScriptSuccess(lines)
1162enddef
1163
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001164def Test_vim9script_call()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001165 var lines =<< trim END
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001166 vim9script
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001167 var name = ''
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001168 def MyFunc(arg: string)
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001169 name = arg
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001170 enddef
1171 MyFunc('foobar')
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001172 name->assert_equal('foobar')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001173
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001174 var str = 'barfoo'
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001175 str->MyFunc()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001176 name->assert_equal('barfoo')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001177
Bram Moolenaar67979662020-06-20 22:50:47 +02001178 g:value = 'value'
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001179 g:value->MyFunc()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001180 name->assert_equal('value')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001181
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001182 var listvar = []
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001183 def ListFunc(arg: list<number>)
1184 listvar = arg
1185 enddef
1186 [1, 2, 3]->ListFunc()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001187 listvar->assert_equal([1, 2, 3])
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001188
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001189 var dictvar = {}
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001190 def DictFunc(arg: dict<number>)
1191 dictvar = arg
1192 enddef
Bram Moolenaare0de1712020-12-02 17:36:54 +01001193 {a: 1, b: 2}->DictFunc()
1194 dictvar->assert_equal({a: 1, b: 2})
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001195 def CompiledDict()
Bram Moolenaare0de1712020-12-02 17:36:54 +01001196 {a: 3, b: 4}->DictFunc()
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001197 enddef
1198 CompiledDict()
Bram Moolenaare0de1712020-12-02 17:36:54 +01001199 dictvar->assert_equal({a: 3, b: 4})
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001200
Bram Moolenaare0de1712020-12-02 17:36:54 +01001201 {a: 3, b: 4}->DictFunc()
1202 dictvar->assert_equal({a: 3, b: 4})
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001203
1204 ('text')->MyFunc()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001205 name->assert_equal('text')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001206 ("some")->MyFunc()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001207 name->assert_equal('some')
Bram Moolenaare6b53242020-07-01 17:28:33 +02001208
Bram Moolenaar13e12b82020-07-24 18:47:22 +02001209 # line starting with single quote is not a mark
Bram Moolenaar10409562020-07-29 20:00:38 +02001210 # line starting with double quote can be a method call
Bram Moolenaar3d48e252020-07-15 14:15:52 +02001211 'asdfasdf'->MyFunc()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001212 name->assert_equal('asdfasdf')
Bram Moolenaar10409562020-07-29 20:00:38 +02001213 "xyz"->MyFunc()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001214 name->assert_equal('xyz')
Bram Moolenaar3d48e252020-07-15 14:15:52 +02001215
1216 def UseString()
1217 'xyork'->MyFunc()
1218 enddef
1219 UseString()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001220 name->assert_equal('xyork')
Bram Moolenaar3d48e252020-07-15 14:15:52 +02001221
Bram Moolenaar10409562020-07-29 20:00:38 +02001222 def UseString2()
1223 "knife"->MyFunc()
1224 enddef
1225 UseString2()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001226 name->assert_equal('knife')
Bram Moolenaar10409562020-07-29 20:00:38 +02001227
Bram Moolenaar13e12b82020-07-24 18:47:22 +02001228 # prepending a colon makes it a mark
1229 new
1230 setline(1, ['aaa', 'bbb', 'ccc'])
1231 normal! 3Gmt1G
1232 :'t
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001233 getcurpos()[1]->assert_equal(3)
Bram Moolenaar13e12b82020-07-24 18:47:22 +02001234 bwipe!
1235
Bram Moolenaare6b53242020-07-01 17:28:33 +02001236 MyFunc(
1237 'continued'
1238 )
1239 assert_equal('continued',
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001240 name
Bram Moolenaare6b53242020-07-01 17:28:33 +02001241 )
1242
1243 call MyFunc(
1244 'more'
1245 ..
1246 'lines'
1247 )
1248 assert_equal(
1249 'morelines',
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001250 name)
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001251 END
1252 writefile(lines, 'Xcall.vim')
1253 source Xcall.vim
1254 delete('Xcall.vim')
1255enddef
1256
1257def Test_vim9script_call_fail_decl()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001258 var lines =<< trim END
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001259 vim9script
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001260 var name = ''
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001261 def MyFunc(arg: string)
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001262 var name = 123
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001263 enddef
Bram Moolenaar822ba242020-05-24 23:00:18 +02001264 defcompile
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001265 END
Bram Moolenaar6c4bfe42020-07-23 18:26:30 +02001266 CheckScriptFailure(lines, 'E1054:')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001267enddef
1268
Bram Moolenaar65b95452020-07-19 14:03:09 +02001269def Test_vim9script_call_fail_type()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001270 var lines =<< trim END
Bram Moolenaar65b95452020-07-19 14:03:09 +02001271 vim9script
1272 def MyFunc(arg: string)
1273 echo arg
1274 enddef
1275 MyFunc(1234)
1276 END
Bram Moolenaar77072282020-09-16 17:55:40 +02001277 CheckScriptFailure(lines, 'E1013: Argument 1: type mismatch, expected string but got number')
Bram Moolenaar65b95452020-07-19 14:03:09 +02001278enddef
1279
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001280def Test_vim9script_call_fail_const()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001281 var lines =<< trim END
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001282 vim9script
1283 const var = ''
1284 def MyFunc(arg: string)
1285 var = 'asdf'
1286 enddef
Bram Moolenaar822ba242020-05-24 23:00:18 +02001287 defcompile
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001288 END
1289 writefile(lines, 'Xcall_const.vim')
Bram Moolenaar9bd5d872020-09-06 21:47:48 +02001290 assert_fails('source Xcall_const.vim', 'E46:', '', 1, 'MyFunc')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001291 delete('Xcall_const.vim')
Bram Moolenaar3bdc90b2020-12-22 20:35:40 +01001292
1293 lines =<< trim END
1294 const g:Aconst = 77
1295 def Change()
1296 # comment
1297 g:Aconst = 99
1298 enddef
1299 call Change()
1300 unlet g:Aconst
1301 END
Bram Moolenaar1dcf55d2020-12-22 22:07:30 +01001302 CheckScriptFailure(lines, 'E741: Value is locked: Aconst', 2)
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001303enddef
1304
1305" Test that inside :function a Python function can be defined, :def is not
1306" recognized.
1307func Test_function_python()
1308 CheckFeature python3
Bram Moolenaar727345e2020-09-27 23:33:59 +02001309 let py = 'python3'
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001310 execute py "<< EOF"
1311def do_something():
1312 return 1
1313EOF
1314endfunc
1315
1316def Test_delfunc()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001317 var lines =<< trim END
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001318 vim9script
Bram Moolenaar4c17ad92020-04-27 22:47:51 +02001319 def g:GoneSoon()
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001320 echo 'hello'
1321 enddef
1322
1323 def CallGoneSoon()
1324 GoneSoon()
1325 enddef
Bram Moolenaar822ba242020-05-24 23:00:18 +02001326 defcompile
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001327
Bram Moolenaar4c17ad92020-04-27 22:47:51 +02001328 delfunc g:GoneSoon
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001329 CallGoneSoon()
1330 END
1331 writefile(lines, 'XToDelFunc')
Bram Moolenaar9bd5d872020-09-06 21:47:48 +02001332 assert_fails('so XToDelFunc', 'E933:', '', 1, 'CallGoneSoon')
1333 assert_fails('so XToDelFunc', 'E933:', '', 1, 'CallGoneSoon')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001334
1335 delete('XToDelFunc')
1336enddef
1337
1338def Test_redef_failure()
Bram Moolenaard2c61702020-09-06 15:58:36 +02001339 writefile(['def Func0(): string', 'return "Func0"', 'enddef'], 'Xdef')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001340 so Xdef
Bram Moolenaard2c61702020-09-06 15:58:36 +02001341 writefile(['def Func1(): string', 'return "Func1"', 'enddef'], 'Xdef')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001342 so Xdef
Bram Moolenaard2c61702020-09-06 15:58:36 +02001343 writefile(['def! Func0(): string', 'enddef', 'defcompile'], 'Xdef')
Bram Moolenaar9bd5d872020-09-06 21:47:48 +02001344 assert_fails('so Xdef', 'E1027:', '', 1, 'Func0')
Bram Moolenaard2c61702020-09-06 15:58:36 +02001345 writefile(['def Func2(): string', 'return "Func2"', 'enddef'], 'Xdef')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001346 so Xdef
Bram Moolenaard2c61702020-09-06 15:58:36 +02001347 delete('Xdef')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001348
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001349 g:Func0()->assert_equal(0)
1350 g:Func1()->assert_equal('Func1')
1351 g:Func2()->assert_equal('Func2')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001352
1353 delfunc! Func0
1354 delfunc! Func1
1355 delfunc! Func2
1356enddef
1357
Bram Moolenaarf93c7fe2020-04-23 22:16:53 +02001358def Test_vim9script_func()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001359 var lines =<< trim END
Bram Moolenaarf93c7fe2020-04-23 22:16:53 +02001360 vim9script
1361 func Func(arg)
1362 echo a:arg
1363 endfunc
1364 Func('text')
1365 END
1366 writefile(lines, 'XVim9Func')
1367 so XVim9Func
1368
1369 delete('XVim9Func')
1370enddef
1371
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001372let s:funcResult = 0
1373
1374def FuncNoArgNoRet()
Bram Moolenaar53900992020-08-22 19:02:02 +02001375 s:funcResult = 11
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001376enddef
1377
1378def FuncNoArgRetNumber(): number
Bram Moolenaar53900992020-08-22 19:02:02 +02001379 s:funcResult = 22
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001380 return 1234
1381enddef
1382
Bram Moolenaarec5929d2020-04-07 20:53:39 +02001383def FuncNoArgRetString(): string
Bram Moolenaar53900992020-08-22 19:02:02 +02001384 s:funcResult = 45
Bram Moolenaarec5929d2020-04-07 20:53:39 +02001385 return 'text'
1386enddef
1387
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001388def FuncOneArgNoRet(arg: number)
Bram Moolenaar53900992020-08-22 19:02:02 +02001389 s:funcResult = arg
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001390enddef
1391
1392def FuncOneArgRetNumber(arg: number): number
Bram Moolenaar53900992020-08-22 19:02:02 +02001393 s:funcResult = arg
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001394 return arg
1395enddef
1396
Bram Moolenaar08938ee2020-04-11 23:17:17 +02001397def FuncTwoArgNoRet(one: bool, two: number)
Bram Moolenaar53900992020-08-22 19:02:02 +02001398 s:funcResult = two
Bram Moolenaar08938ee2020-04-11 23:17:17 +02001399enddef
1400
Bram Moolenaarec5929d2020-04-07 20:53:39 +02001401def FuncOneArgRetString(arg: string): string
1402 return arg
1403enddef
1404
Bram Moolenaar89228602020-04-05 22:14:54 +02001405def FuncOneArgRetAny(arg: any): any
1406 return arg
1407enddef
1408
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001409def Test_func_type()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001410 var Ref1: func()
Bram Moolenaar53900992020-08-22 19:02:02 +02001411 s:funcResult = 0
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001412 Ref1 = FuncNoArgNoRet
1413 Ref1()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001414 s:funcResult->assert_equal(11)
Bram Moolenaar4c683752020-04-05 21:38:23 +02001415
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001416 var Ref2: func
Bram Moolenaar53900992020-08-22 19:02:02 +02001417 s:funcResult = 0
Bram Moolenaar4c683752020-04-05 21:38:23 +02001418 Ref2 = FuncNoArgNoRet
1419 Ref2()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001420 s:funcResult->assert_equal(11)
Bram Moolenaar4c683752020-04-05 21:38:23 +02001421
Bram Moolenaar53900992020-08-22 19:02:02 +02001422 s:funcResult = 0
Bram Moolenaar4c683752020-04-05 21:38:23 +02001423 Ref2 = FuncOneArgNoRet
1424 Ref2(12)
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001425 s:funcResult->assert_equal(12)
Bram Moolenaar4c683752020-04-05 21:38:23 +02001426
Bram Moolenaar53900992020-08-22 19:02:02 +02001427 s:funcResult = 0
Bram Moolenaar4c683752020-04-05 21:38:23 +02001428 Ref2 = FuncNoArgRetNumber
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001429 Ref2()->assert_equal(1234)
1430 s:funcResult->assert_equal(22)
Bram Moolenaar4c683752020-04-05 21:38:23 +02001431
Bram Moolenaar53900992020-08-22 19:02:02 +02001432 s:funcResult = 0
Bram Moolenaar4c683752020-04-05 21:38:23 +02001433 Ref2 = FuncOneArgRetNumber
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001434 Ref2(13)->assert_equal(13)
1435 s:funcResult->assert_equal(13)
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001436enddef
1437
Bram Moolenaar9978d472020-07-05 16:01:56 +02001438def Test_repeat_return_type()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001439 var res = 0
Bram Moolenaar9978d472020-07-05 16:01:56 +02001440 for n in repeat([1], 3)
1441 res += n
1442 endfor
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001443 res->assert_equal(3)
Bram Moolenaarfce82b32020-07-05 16:07:21 +02001444
1445 res = 0
1446 for n in add([1, 2], 3)
1447 res += n
1448 endfor
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001449 res->assert_equal(6)
Bram Moolenaar9978d472020-07-05 16:01:56 +02001450enddef
1451
Bram Moolenaar846178a2020-07-05 17:04:13 +02001452def Test_argv_return_type()
1453 next fileone filetwo
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001454 var res = ''
Bram Moolenaar846178a2020-07-05 17:04:13 +02001455 for name in argv()
1456 res ..= name
1457 endfor
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001458 res->assert_equal('fileonefiletwo')
Bram Moolenaar846178a2020-07-05 17:04:13 +02001459enddef
1460
Bram Moolenaarec5929d2020-04-07 20:53:39 +02001461def Test_func_type_part()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001462 var RefVoid: func: void
Bram Moolenaarec5929d2020-04-07 20:53:39 +02001463 RefVoid = FuncNoArgNoRet
1464 RefVoid = FuncOneArgNoRet
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001465 CheckDefFailure(['var RefVoid: func: void', 'RefVoid = FuncNoArgRetNumber'], 'E1012: Type mismatch; expected func(...) but got func(): number')
1466 CheckDefFailure(['var RefVoid: func: void', 'RefVoid = FuncNoArgRetString'], 'E1012: Type mismatch; expected func(...) but got func(): string')
Bram Moolenaarec5929d2020-04-07 20:53:39 +02001467
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001468 var RefAny: func(): any
Bram Moolenaarec5929d2020-04-07 20:53:39 +02001469 RefAny = FuncNoArgRetNumber
1470 RefAny = FuncNoArgRetString
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001471 CheckDefFailure(['var RefAny: func(): any', 'RefAny = FuncNoArgNoRet'], 'E1012: Type mismatch; expected func(): any but got func()')
1472 CheckDefFailure(['var RefAny: func(): any', 'RefAny = FuncOneArgNoRet'], 'E1012: Type mismatch; expected func(): any but got func(number)')
Bram Moolenaarec5929d2020-04-07 20:53:39 +02001473
Bram Moolenaar6abd3dc2020-10-04 14:17:32 +02001474 var RefAnyNoArgs: func: any = RefAny
1475
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001476 var RefNr: func: number
Bram Moolenaarec5929d2020-04-07 20:53:39 +02001477 RefNr = FuncNoArgRetNumber
1478 RefNr = FuncOneArgRetNumber
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001479 CheckDefFailure(['var RefNr: func: number', 'RefNr = FuncNoArgNoRet'], 'E1012: Type mismatch; expected func(...): number but got func()')
1480 CheckDefFailure(['var RefNr: func: number', 'RefNr = FuncNoArgRetString'], 'E1012: Type mismatch; expected func(...): number but got func(): string')
Bram Moolenaarec5929d2020-04-07 20:53:39 +02001481
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001482 var RefStr: func: string
Bram Moolenaarec5929d2020-04-07 20:53:39 +02001483 RefStr = FuncNoArgRetString
1484 RefStr = FuncOneArgRetString
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001485 CheckDefFailure(['var RefStr: func: string', 'RefStr = FuncNoArgNoRet'], 'E1012: Type mismatch; expected func(...): string but got func()')
1486 CheckDefFailure(['var RefStr: func: string', 'RefStr = FuncNoArgRetNumber'], 'E1012: Type mismatch; expected func(...): string but got func(): number')
Bram Moolenaarec5929d2020-04-07 20:53:39 +02001487enddef
1488
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001489def Test_func_type_fails()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001490 CheckDefFailure(['var ref1: func()'], 'E704:')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001491
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001492 CheckDefFailure(['var Ref1: func()', 'Ref1 = FuncNoArgRetNumber'], 'E1012: Type mismatch; expected func() but got func(): number')
1493 CheckDefFailure(['var Ref1: func()', 'Ref1 = FuncOneArgNoRet'], 'E1012: Type mismatch; expected func() but got func(number)')
1494 CheckDefFailure(['var Ref1: func()', 'Ref1 = FuncOneArgRetNumber'], 'E1012: Type mismatch; expected func() but got func(number): number')
1495 CheckDefFailure(['var Ref1: func(bool)', 'Ref1 = FuncTwoArgNoRet'], 'E1012: Type mismatch; expected func(bool) but got func(bool, number)')
1496 CheckDefFailure(['var Ref1: func(?bool)', 'Ref1 = FuncTwoArgNoRet'], 'E1012: Type mismatch; expected func(?bool) but got func(bool, number)')
1497 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 +02001498
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001499 CheckDefFailure(['var RefWrong: func(string ,number)'], 'E1068:')
1500 CheckDefFailure(['var RefWrong: func(string,number)'], 'E1069:')
1501 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:')
1502 CheckDefFailure(['var RefWrong: func(bool):string'], 'E1069:')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001503enddef
1504
Bram Moolenaar89228602020-04-05 22:14:54 +02001505def Test_func_return_type()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001506 var nr: number
Bram Moolenaar89228602020-04-05 22:14:54 +02001507 nr = FuncNoArgRetNumber()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001508 nr->assert_equal(1234)
Bram Moolenaar89228602020-04-05 22:14:54 +02001509
1510 nr = FuncOneArgRetAny(122)
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001511 nr->assert_equal(122)
Bram Moolenaar89228602020-04-05 22:14:54 +02001512
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001513 var str: string
Bram Moolenaar89228602020-04-05 22:14:54 +02001514 str = FuncOneArgRetAny('yes')
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001515 str->assert_equal('yes')
Bram Moolenaar89228602020-04-05 22:14:54 +02001516
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001517 CheckDefFailure(['var str: string', 'str = FuncNoArgRetNumber()'], 'E1012: Type mismatch; expected string but got number')
Bram Moolenaar89228602020-04-05 22:14:54 +02001518enddef
1519
Bram Moolenaar6abd3dc2020-10-04 14:17:32 +02001520def Test_func_common_type()
1521 def FuncOne(n: number): number
1522 return n
1523 enddef
1524 def FuncTwo(s: string): number
1525 return len(s)
1526 enddef
1527 def FuncThree(n: number, s: string): number
1528 return n + len(s)
1529 enddef
1530 var list = [FuncOne, FuncTwo, FuncThree]
1531 assert_equal(8, list[0](8))
1532 assert_equal(4, list[1]('word'))
1533 assert_equal(7, list[2](3, 'word'))
1534enddef
1535
Bram Moolenaar5e774c72020-04-12 21:53:00 +02001536def MultiLine(
1537 arg1: string,
1538 arg2 = 1234,
1539 ...rest: list<string>
1540 ): string
1541 return arg1 .. arg2 .. join(rest, '-')
1542enddef
1543
Bram Moolenaar2c330432020-04-13 14:41:35 +02001544def MultiLineComment(
1545 arg1: string, # comment
1546 arg2 = 1234, # comment
1547 ...rest: list<string> # comment
1548 ): string # comment
1549 return arg1 .. arg2 .. join(rest, '-')
1550enddef
1551
Bram Moolenaar5e774c72020-04-12 21:53:00 +02001552def Test_multiline()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001553 MultiLine('text')->assert_equal('text1234')
1554 MultiLine('text', 777)->assert_equal('text777')
1555 MultiLine('text', 777, 'one')->assert_equal('text777one')
1556 MultiLine('text', 777, 'one', 'two')->assert_equal('text777one-two')
Bram Moolenaar5e774c72020-04-12 21:53:00 +02001557enddef
1558
Bram Moolenaar23e03252020-04-12 22:22:31 +02001559func Test_multiline_not_vim9()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001560 call MultiLine('text')->assert_equal('text1234')
1561 call MultiLine('text', 777)->assert_equal('text777')
1562 call MultiLine('text', 777, 'one')->assert_equal('text777one')
1563 call MultiLine('text', 777, 'one', 'two')->assert_equal('text777one-two')
Bram Moolenaar23e03252020-04-12 22:22:31 +02001564endfunc
1565
Bram Moolenaar5e774c72020-04-12 21:53:00 +02001566
Bram Moolenaaree4e0c12020-04-06 21:35:05 +02001567" When using CheckScriptFailure() for the below test, E1010 is generated instead
1568" of E1056.
1569func Test_E1056_1059()
1570 let caught_1056 = 0
1571 try
1572 def F():
1573 return 1
1574 enddef
1575 catch /E1056:/
1576 let caught_1056 = 1
1577 endtry
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001578 eval caught_1056->assert_equal(1)
Bram Moolenaaree4e0c12020-04-06 21:35:05 +02001579
1580 let caught_1059 = 0
1581 try
1582 def F5(items : list)
1583 echo 'a'
1584 enddef
1585 catch /E1059:/
1586 let caught_1059 = 1
1587 endtry
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001588 eval caught_1059->assert_equal(1)
Bram Moolenaaree4e0c12020-04-06 21:35:05 +02001589endfunc
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001590
Bram Moolenaar015f4262020-05-05 21:25:22 +02001591func DelMe()
1592 echo 'DelMe'
1593endfunc
1594
Bram Moolenaarbf8feb52020-08-08 14:26:31 +02001595def Test_error_reporting()
1596 # comment lines at the start of the function
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001597 var lines =<< trim END
Bram Moolenaarbf8feb52020-08-08 14:26:31 +02001598 " comment
1599 def Func()
1600 # comment
1601 # comment
1602 invalid
1603 enddef
1604 defcompile
1605 END
Bram Moolenaar08052222020-09-14 17:04:31 +02001606 writefile(lines, 'Xdef')
Bram Moolenaarbf8feb52020-08-08 14:26:31 +02001607 try
1608 source Xdef
Bram Moolenaar7517ffd2020-08-14 18:35:07 +02001609 assert_report('should have failed')
Bram Moolenaarbf8feb52020-08-08 14:26:31 +02001610 catch /E476:/
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001611 v:exception->assert_match('Invalid command: invalid')
1612 v:throwpoint->assert_match(', line 3$')
Bram Moolenaarbf8feb52020-08-08 14:26:31 +02001613 endtry
Bram Moolenaar2d870f82020-12-05 13:41:01 +01001614 delfunc! g:Func
Bram Moolenaarbf8feb52020-08-08 14:26:31 +02001615
1616 # comment lines after the start of the function
1617 lines =<< trim END
1618 " comment
1619 def Func()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001620 var x = 1234
Bram Moolenaarbf8feb52020-08-08 14:26:31 +02001621 # comment
1622 # comment
1623 invalid
1624 enddef
1625 defcompile
1626 END
Bram Moolenaar08052222020-09-14 17:04:31 +02001627 writefile(lines, 'Xdef')
Bram Moolenaarbf8feb52020-08-08 14:26:31 +02001628 try
1629 source Xdef
Bram Moolenaar7517ffd2020-08-14 18:35:07 +02001630 assert_report('should have failed')
Bram Moolenaarbf8feb52020-08-08 14:26:31 +02001631 catch /E476:/
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001632 v:exception->assert_match('Invalid command: invalid')
1633 v:throwpoint->assert_match(', line 4$')
Bram Moolenaarbf8feb52020-08-08 14:26:31 +02001634 endtry
Bram Moolenaar2d870f82020-12-05 13:41:01 +01001635 delfunc! g:Func
Bram Moolenaarbf8feb52020-08-08 14:26:31 +02001636
Bram Moolenaar7517ffd2020-08-14 18:35:07 +02001637 lines =<< trim END
1638 vim9script
1639 def Func()
Bram Moolenaare0de1712020-12-02 17:36:54 +01001640 var db = {foo: 1, bar: 2}
Bram Moolenaar7517ffd2020-08-14 18:35:07 +02001641 # comment
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001642 var x = db.asdf
Bram Moolenaar7517ffd2020-08-14 18:35:07 +02001643 enddef
1644 defcompile
1645 Func()
1646 END
Bram Moolenaar08052222020-09-14 17:04:31 +02001647 writefile(lines, 'Xdef')
Bram Moolenaar7517ffd2020-08-14 18:35:07 +02001648 try
1649 source Xdef
1650 assert_report('should have failed')
1651 catch /E716:/
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001652 v:throwpoint->assert_match('_Func, line 3$')
Bram Moolenaar7517ffd2020-08-14 18:35:07 +02001653 endtry
Bram Moolenaar2d870f82020-12-05 13:41:01 +01001654 delfunc! g:Func
Bram Moolenaar7517ffd2020-08-14 18:35:07 +02001655
Bram Moolenaar08052222020-09-14 17:04:31 +02001656 delete('Xdef')
Bram Moolenaarbf8feb52020-08-08 14:26:31 +02001657enddef
1658
Bram Moolenaar015f4262020-05-05 21:25:22 +02001659def Test_deleted_function()
1660 CheckDefExecFailure([
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001661 'var RefMe: func = function("g:DelMe")',
Bram Moolenaar015f4262020-05-05 21:25:22 +02001662 'delfunc g:DelMe',
1663 'echo RefMe()'], 'E117:')
1664enddef
1665
1666def Test_unknown_function()
1667 CheckDefExecFailure([
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001668 'var Ref: func = function("NotExist")',
Bram Moolenaar9b7bf9e2020-07-11 22:14:59 +02001669 'delfunc g:NotExist'], 'E700:')
Bram Moolenaar015f4262020-05-05 21:25:22 +02001670enddef
1671
Bram Moolenaar328eac22021-01-07 19:23:08 +01001672def RefFunc(Ref: func(any): any): string
Bram Moolenaarc8cd2b32020-05-01 19:29:08 +02001673 return Ref('more')
1674enddef
1675
1676def Test_closure_simple()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001677 var local = 'some '
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01001678 RefFunc((s) => local .. s)->assert_equal('some more')
Bram Moolenaarc8cd2b32020-05-01 19:29:08 +02001679enddef
1680
Bram Moolenaarbf67ea12020-05-02 17:52:42 +02001681def MakeRef()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001682 var local = 'some '
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01001683 g:Ref = (s) => local .. s
Bram Moolenaarbf67ea12020-05-02 17:52:42 +02001684enddef
1685
1686def Test_closure_ref_after_return()
1687 MakeRef()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001688 g:Ref('thing')->assert_equal('some thing')
Bram Moolenaarbf67ea12020-05-02 17:52:42 +02001689 unlet g:Ref
1690enddef
1691
Bram Moolenaar5adc55c2020-05-02 23:12:58 +02001692def MakeTwoRefs()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001693 var local = ['some']
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01001694 g:Extend = (s) => local->add(s)
1695 g:Read = () => local
Bram Moolenaar5adc55c2020-05-02 23:12:58 +02001696enddef
1697
1698def Test_closure_two_refs()
1699 MakeTwoRefs()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001700 join(g:Read(), ' ')->assert_equal('some')
Bram Moolenaar5adc55c2020-05-02 23:12:58 +02001701 g:Extend('more')
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001702 join(g:Read(), ' ')->assert_equal('some more')
Bram Moolenaar5adc55c2020-05-02 23:12:58 +02001703 g:Extend('even')
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001704 join(g:Read(), ' ')->assert_equal('some more even')
Bram Moolenaar5adc55c2020-05-02 23:12:58 +02001705
1706 unlet g:Extend
1707 unlet g:Read
1708enddef
1709
Bram Moolenaar5adc55c2020-05-02 23:12:58 +02001710def ReadRef(Ref: func(): list<string>): string
1711 return join(Ref(), ' ')
1712enddef
1713
Bram Moolenaar5e654232020-09-16 15:22:00 +02001714def ExtendRef(Ref: func(string): list<string>, add: string)
Bram Moolenaar5adc55c2020-05-02 23:12:58 +02001715 Ref(add)
1716enddef
1717
1718def Test_closure_two_indirect_refs()
Bram Moolenaarf7779c62020-05-03 15:38:16 +02001719 MakeTwoRefs()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001720 ReadRef(g:Read)->assert_equal('some')
Bram Moolenaar5adc55c2020-05-02 23:12:58 +02001721 ExtendRef(g:Extend, 'more')
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001722 ReadRef(g:Read)->assert_equal('some more')
Bram Moolenaar5adc55c2020-05-02 23:12:58 +02001723 ExtendRef(g:Extend, 'even')
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001724 ReadRef(g:Read)->assert_equal('some more even')
Bram Moolenaar5adc55c2020-05-02 23:12:58 +02001725
1726 unlet g:Extend
1727 unlet g:Read
1728enddef
Bram Moolenaarbf67ea12020-05-02 17:52:42 +02001729
Bram Moolenaar2fd4cd72020-05-03 22:30:49 +02001730def MakeArgRefs(theArg: string)
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001731 var local = 'loc_val'
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01001732 g:UseArg = (s) => theArg .. '/' .. local .. '/' .. s
Bram Moolenaar2fd4cd72020-05-03 22:30:49 +02001733enddef
1734
1735def MakeArgRefsVarargs(theArg: string, ...rest: list<string>)
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001736 var local = 'the_loc'
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01001737 g:UseVararg = (s) => theArg .. '/' .. local .. '/' .. s .. '/' .. join(rest)
Bram Moolenaar2fd4cd72020-05-03 22:30:49 +02001738enddef
1739
1740def Test_closure_using_argument()
1741 MakeArgRefs('arg_val')
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001742 g:UseArg('call_val')->assert_equal('arg_val/loc_val/call_val')
Bram Moolenaar2fd4cd72020-05-03 22:30:49 +02001743
1744 MakeArgRefsVarargs('arg_val', 'one', 'two')
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001745 g:UseVararg('call_val')->assert_equal('arg_val/the_loc/call_val/one two')
Bram Moolenaar2fd4cd72020-05-03 22:30:49 +02001746
1747 unlet g:UseArg
1748 unlet g:UseVararg
Bram Moolenaar44ec21c2021-02-12 21:50:57 +01001749
1750 var lines =<< trim END
1751 vim9script
1752 def Test(Fun: func(number): number): list<number>
1753 return map([1, 2, 3], (_, i) => Fun(i))
1754 enddef
1755 def Inc(nr: number): number
1756 return nr + 2
1757 enddef
1758 assert_equal([3, 4, 5], Test(Inc))
1759 END
1760 CheckScriptSuccess(lines)
Bram Moolenaar2fd4cd72020-05-03 22:30:49 +02001761enddef
1762
Bram Moolenaar85d5e2b2020-10-10 14:13:01 +02001763def MakeGetAndAppendRefs()
1764 var local = 'a'
1765
1766 def Append(arg: string)
1767 local ..= arg
1768 enddef
1769 g:Append = Append
1770
1771 def Get(): string
1772 return local
1773 enddef
1774 g:Get = Get
1775enddef
1776
1777def Test_closure_append_get()
1778 MakeGetAndAppendRefs()
1779 g:Get()->assert_equal('a')
1780 g:Append('-b')
1781 g:Get()->assert_equal('a-b')
1782 g:Append('-c')
1783 g:Get()->assert_equal('a-b-c')
1784
1785 unlet g:Append
1786 unlet g:Get
1787enddef
Bram Moolenaarb68b3462020-05-06 21:06:30 +02001788
Bram Moolenaar04b12692020-05-04 23:24:44 +02001789def Test_nested_closure()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001790 var local = 'text'
Bram Moolenaar04b12692020-05-04 23:24:44 +02001791 def Closure(arg: string): string
1792 return local .. arg
1793 enddef
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001794 Closure('!!!')->assert_equal('text!!!')
Bram Moolenaar04b12692020-05-04 23:24:44 +02001795enddef
1796
Bram Moolenaar6f5b6df2020-05-16 21:20:12 +02001797func GetResult(Ref)
1798 return a:Ref('some')
1799endfunc
1800
1801def Test_call_closure_not_compiled()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001802 var text = 'text'
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01001803 g:Ref = (s) => s .. text
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001804 GetResult(g:Ref)->assert_equal('sometext')
Bram Moolenaar6f5b6df2020-05-16 21:20:12 +02001805enddef
1806
Bram Moolenaar7cbfaa52020-09-18 21:25:32 +02001807def Test_double_closure_fails()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001808 var lines =<< trim END
Bram Moolenaar7cbfaa52020-09-18 21:25:32 +02001809 vim9script
1810 def Func()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001811 var name = 0
1812 for i in range(2)
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01001813 timer_start(0, () => name)
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001814 endfor
Bram Moolenaar7cbfaa52020-09-18 21:25:32 +02001815 enddef
1816 Func()
1817 END
Bram Moolenaar148ce7a2020-09-23 21:57:23 +02001818 CheckScriptSuccess(lines)
Bram Moolenaar7cbfaa52020-09-18 21:25:32 +02001819enddef
1820
Bram Moolenaar85d5e2b2020-10-10 14:13:01 +02001821def Test_nested_closure_used()
1822 var lines =<< trim END
1823 vim9script
1824 def Func()
1825 var x = 'hello'
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01001826 var Closure = () => x
1827 g:Myclosure = () => Closure()
Bram Moolenaar85d5e2b2020-10-10 14:13:01 +02001828 enddef
1829 Func()
1830 assert_equal('hello', g:Myclosure())
1831 END
1832 CheckScriptSuccess(lines)
1833enddef
Bram Moolenaar0876c782020-10-07 19:08:04 +02001834
Bram Moolenaarc70bdab2020-09-26 19:59:38 +02001835def Test_nested_closure_fails()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001836 var lines =<< trim END
Bram Moolenaarc70bdab2020-09-26 19:59:38 +02001837 vim9script
1838 def FuncA()
1839 FuncB(0)
1840 enddef
1841 def FuncB(n: number): list<string>
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01001842 return map([0], (_, v) => n)
Bram Moolenaarc70bdab2020-09-26 19:59:38 +02001843 enddef
1844 FuncA()
1845 END
1846 CheckScriptFailure(lines, 'E1012:')
1847enddef
1848
Bram Moolenaarf112f302020-12-20 17:47:52 +01001849def Test_global_closure()
1850 var lines =<< trim END
1851 vim9script
1852 def ReverseEveryNLines(n: number, line1: number, line2: number)
1853 var mods = 'sil keepj keepp lockm '
1854 var range = ':' .. line1 .. ',' .. line2
1855 def g:Offset(): number
1856 var offset = (line('.') - line1 + 1) % n
1857 return offset != 0 ? offset : n
1858 enddef
1859 exe mods .. range .. 'g/^/exe "m .-" .. g:Offset()'
1860 enddef
1861
1862 new
1863 repeat(['aaa', 'bbb', 'ccc'], 3)->setline(1)
1864 ReverseEveryNLines(3, 1, 9)
1865 END
1866 CheckScriptSuccess(lines)
1867 var expected = repeat(['ccc', 'bbb', 'aaa'], 3)
1868 assert_equal(expected, getline(1, 9))
1869 bwipe!
1870enddef
1871
Bram Moolenaarcd45ed02020-12-22 17:35:54 +01001872def Test_global_closure_called_directly()
1873 var lines =<< trim END
1874 vim9script
1875 def Outer()
1876 var x = 1
1877 def g:Inner()
1878 var y = x
1879 x += 1
1880 assert_equal(1, y)
1881 enddef
1882 g:Inner()
1883 assert_equal(2, x)
1884 enddef
1885 Outer()
1886 END
1887 CheckScriptSuccess(lines)
1888 delfunc g:Inner
1889enddef
1890
Bram Moolenaar34c54eb2020-11-25 19:15:19 +01001891def Test_failure_in_called_function()
1892 # this was using the frame index as the return value
1893 var lines =<< trim END
1894 vim9script
1895 au TerminalWinOpen * eval [][0]
1896 def PopupTerm(a: any)
1897 # make sure typvals on stack are string
1898 ['a', 'b', 'c', 'd', 'e', 'f', 'g']->join()
1899 FireEvent()
1900 enddef
1901 def FireEvent()
1902 do TerminalWinOpen
1903 enddef
1904 # use try/catch to make eval fail
1905 try
1906 call PopupTerm(0)
1907 catch
1908 endtry
1909 au! TerminalWinOpen
1910 END
1911 CheckScriptSuccess(lines)
1912enddef
1913
Bram Moolenaar5366e1a2020-10-01 13:01:34 +02001914def Test_nested_lambda()
1915 var lines =<< trim END
1916 vim9script
1917 def Func()
1918 var x = 4
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01001919 var Lambda1 = () => 7
1920 var Lambda2 = () => [Lambda1(), x]
Bram Moolenaar5366e1a2020-10-01 13:01:34 +02001921 var res = Lambda2()
1922 assert_equal([7, 4], res)
1923 enddef
1924 Func()
1925 END
1926 CheckScriptSuccess(lines)
1927enddef
1928
Bram Moolenaar52bf81c2020-11-17 18:50:44 +01001929def Shadowed(): list<number>
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01001930 var FuncList: list<func: number> = [() => 42]
Bram Moolenaar75ab91f2021-01-10 22:42:50 +01001931 return FuncList->mapnew((_, Shadowed) => Shadowed())
Bram Moolenaar52bf81c2020-11-17 18:50:44 +01001932enddef
1933
1934def Test_lambda_arg_shadows_func()
1935 assert_equal([42], Shadowed())
1936enddef
1937
Bram Moolenaaracd4c5e2020-06-22 19:39:03 +02001938def Line_continuation_in_def(dir: string = ''): string
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001939 var path: string = empty(dir)
1940 \ ? 'empty'
1941 \ : 'full'
1942 return path
Bram Moolenaaracd4c5e2020-06-22 19:39:03 +02001943enddef
1944
1945def Test_line_continuation_in_def()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001946 Line_continuation_in_def('.')->assert_equal('full')
Bram Moolenaaracd4c5e2020-06-22 19:39:03 +02001947enddef
1948
Bram Moolenaar2ea95b62020-11-19 21:47:56 +01001949def Test_script_var_in_lambda()
1950 var lines =<< trim END
1951 vim9script
1952 var script = 'test'
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01001953 assert_equal(['test'], map(['one'], () => script))
Bram Moolenaar2ea95b62020-11-19 21:47:56 +01001954 END
1955 CheckScriptSuccess(lines)
1956enddef
1957
Bram Moolenaar5e654232020-09-16 15:22:00 +02001958def Line_continuation_in_lambda(): list<string>
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001959 var x = range(97, 100)
Bram Moolenaar75ab91f2021-01-10 22:42:50 +01001960 ->mapnew((_, v) => nr2char(v)
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01001961 ->toupper())
Bram Moolenaar7a4b8982020-07-08 17:36:21 +02001962 ->reverse()
1963 return x
1964enddef
1965
1966def Test_line_continuation_in_lambda()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001967 Line_continuation_in_lambda()->assert_equal(['D', 'C', 'B', 'A'])
Bram Moolenaarf898f7c2021-01-16 18:09:52 +01001968
1969 var lines =<< trim END
1970 vim9script
1971 var res = [{n: 1, m: 2, s: 'xxx'}]
1972 ->mapnew((_, v: dict<any>): string => printf('%d:%d:%s',
1973 v.n,
1974 v.m,
1975 substitute(v.s, '.*', 'yyy', '')
1976 ))
1977 assert_equal(['1:2:yyy'], res)
1978 END
1979 CheckScriptSuccess(lines)
Bram Moolenaar7a4b8982020-07-08 17:36:21 +02001980enddef
1981
Bram Moolenaarb6571982021-01-08 22:24:19 +01001982def Test_list_lambda()
1983 timer_start(1000, (_) => 0)
1984 var body = execute(timer_info()[0].callback
1985 ->string()
1986 ->substitute("('", ' ', '')
1987 ->substitute("')", '', '')
1988 ->substitute('function\zs', ' ', ''))
1989 assert_match('def <lambda>\d\+(_: any, ...): number\n1 return 0\n enddef', body)
1990enddef
1991
Bram Moolenaarab360522021-01-10 14:02:28 +01001992def DoFilterThis(a: string): list<string>
1993 # closure nested inside another closure using argument
1994 var Filter = (l) => filter(l, (_, v) => stridx(v, a) == 0)
1995 return ['x', 'y', 'a', 'x2', 'c']->Filter()
1996enddef
1997
1998def Test_nested_closure_using_argument()
1999 assert_equal(['x', 'x2'], DoFilterThis('x'))
2000enddef
2001
Bram Moolenaar0186e582021-01-10 18:33:11 +01002002def Test_triple_nested_closure()
2003 var what = 'x'
2004 var Match = (val: string, cmp: string): bool => stridx(val, cmp) == 0
2005 var Filter = (l) => filter(l, (_, v) => Match(v, what))
2006 assert_equal(['x', 'x2'], ['x', 'y', 'a', 'x2', 'c']->Filter())
2007enddef
2008
Bram Moolenaar8f510af2020-07-05 18:48:23 +02002009func Test_silent_echo()
Bram Moolenaar47e7d702020-07-05 18:18:42 +02002010 CheckScreendump
2011
2012 let lines =<< trim END
2013 vim9script
2014 def EchoNothing()
2015 silent echo ''
2016 enddef
2017 defcompile
2018 END
Bram Moolenaar8f510af2020-07-05 18:48:23 +02002019 call writefile(lines, 'XTest_silent_echo')
Bram Moolenaar47e7d702020-07-05 18:18:42 +02002020
2021 " Check that the balloon shows up after a mouse move
2022 let buf = RunVimInTerminal('-S XTest_silent_echo', {'rows': 6})
Bram Moolenaar8f510af2020-07-05 18:48:23 +02002023 call term_sendkeys(buf, ":abc")
Bram Moolenaar47e7d702020-07-05 18:18:42 +02002024 call VerifyScreenDump(buf, 'Test_vim9_silent_echo', {})
2025
2026 " clean up
2027 call StopVimInTerminal(buf)
2028 call delete('XTest_silent_echo')
Bram Moolenaar8f510af2020-07-05 18:48:23 +02002029endfunc
Bram Moolenaar47e7d702020-07-05 18:18:42 +02002030
Bram Moolenaar171fb922020-10-28 16:54:47 +01002031def SilentlyError()
2032 execute('silent! invalid')
2033 g:did_it = 'yes'
2034enddef
2035
Bram Moolenaar28ee8922020-10-28 20:20:00 +01002036func UserError()
2037 silent! invalid
2038endfunc
2039
2040def SilentlyUserError()
2041 UserError()
2042 g:did_it = 'yes'
2043enddef
Bram Moolenaar171fb922020-10-28 16:54:47 +01002044
2045" This can't be a :def function, because the assert would not be reached.
Bram Moolenaar171fb922020-10-28 16:54:47 +01002046func Test_ignore_silent_error()
2047 let g:did_it = 'no'
2048 call SilentlyError()
2049 call assert_equal('yes', g:did_it)
2050
Bram Moolenaar28ee8922020-10-28 20:20:00 +01002051 let g:did_it = 'no'
2052 call SilentlyUserError()
2053 call assert_equal('yes', g:did_it)
Bram Moolenaar171fb922020-10-28 16:54:47 +01002054
2055 unlet g:did_it
2056endfunc
2057
Bram Moolenaarcd030c42020-10-30 21:49:40 +01002058def Test_ignore_silent_error_in_filter()
2059 var lines =<< trim END
2060 vim9script
2061 def Filter(winid: number, key: string): bool
2062 if key == 'o'
2063 silent! eval [][0]
2064 return true
2065 endif
2066 return popup_filter_menu(winid, key)
2067 enddef
2068
Bram Moolenaare0de1712020-12-02 17:36:54 +01002069 popup_create('popup', {filter: Filter})
Bram Moolenaarcd030c42020-10-30 21:49:40 +01002070 feedkeys("o\r", 'xnt')
2071 END
2072 CheckScriptSuccess(lines)
2073enddef
2074
Bram Moolenaar4b9bd692020-09-05 21:57:53 +02002075def Fibonacci(n: number): number
2076 if n < 2
2077 return n
2078 else
2079 return Fibonacci(n - 1) + Fibonacci(n - 2)
2080 endif
2081enddef
2082
Bram Moolenaar985116a2020-07-12 17:31:09 +02002083def Test_recursive_call()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002084 Fibonacci(20)->assert_equal(6765)
Bram Moolenaar985116a2020-07-12 17:31:09 +02002085enddef
2086
Bram Moolenaar08f7a412020-07-13 20:41:08 +02002087def TreeWalk(dir: string): list<any>
Bram Moolenaar75ab91f2021-01-10 22:42:50 +01002088 return readdir(dir)->mapnew((_, val) =>
Bram Moolenaar08f7a412020-07-13 20:41:08 +02002089 fnamemodify(dir .. '/' .. val, ':p')->isdirectory()
Bram Moolenaar2bede172020-11-19 18:53:18 +01002090 ? {[val]: TreeWalk(dir .. '/' .. val)}
Bram Moolenaar08f7a412020-07-13 20:41:08 +02002091 : val
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01002092 )
Bram Moolenaar08f7a412020-07-13 20:41:08 +02002093enddef
2094
2095def Test_closure_in_map()
2096 mkdir('XclosureDir/tdir', 'p')
2097 writefile(['111'], 'XclosureDir/file1')
2098 writefile(['222'], 'XclosureDir/file2')
2099 writefile(['333'], 'XclosureDir/tdir/file3')
2100
Bram Moolenaare0de1712020-12-02 17:36:54 +01002101 TreeWalk('XclosureDir')->assert_equal(['file1', 'file2', {tdir: ['file3']}])
Bram Moolenaar08f7a412020-07-13 20:41:08 +02002102
2103 delete('XclosureDir', 'rf')
2104enddef
2105
Bram Moolenaar7b5d5442020-10-04 13:42:34 +02002106def Test_invalid_function_name()
2107 var lines =<< trim END
2108 vim9script
2109 def s: list<string>
2110 END
2111 CheckScriptFailure(lines, 'E129:')
2112
2113 lines =<< trim END
2114 vim9script
2115 def g: list<string>
2116 END
2117 CheckScriptFailure(lines, 'E129:')
2118
2119 lines =<< trim END
2120 vim9script
2121 def <SID>: list<string>
2122 END
2123 CheckScriptFailure(lines, 'E884:')
2124
2125 lines =<< trim END
2126 vim9script
2127 def F list<string>
2128 END
2129 CheckScriptFailure(lines, 'E488:')
2130enddef
2131
Bram Moolenaara90afb92020-07-15 22:38:56 +02002132def Test_partial_call()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002133 var Xsetlist = function('setloclist', [0])
Bram Moolenaare0de1712020-12-02 17:36:54 +01002134 Xsetlist([], ' ', {title: 'test'})
2135 getloclist(0, {title: 1})->assert_equal({title: 'test'})
Bram Moolenaara90afb92020-07-15 22:38:56 +02002136
2137 Xsetlist = function('setloclist', [0, [], ' '])
Bram Moolenaare0de1712020-12-02 17:36:54 +01002138 Xsetlist({title: 'test'})
2139 getloclist(0, {title: 1})->assert_equal({title: 'test'})
Bram Moolenaara90afb92020-07-15 22:38:56 +02002140
2141 Xsetlist = function('setqflist')
Bram Moolenaare0de1712020-12-02 17:36:54 +01002142 Xsetlist([], ' ', {title: 'test'})
2143 getqflist({title: 1})->assert_equal({title: 'test'})
Bram Moolenaara90afb92020-07-15 22:38:56 +02002144
2145 Xsetlist = function('setqflist', [[], ' '])
Bram Moolenaare0de1712020-12-02 17:36:54 +01002146 Xsetlist({title: 'test'})
2147 getqflist({title: 1})->assert_equal({title: 'test'})
Bram Moolenaar6abd3dc2020-10-04 14:17:32 +02002148
2149 var Len: func: number = function('len', ['word'])
2150 assert_equal(4, Len())
Bram Moolenaara90afb92020-07-15 22:38:56 +02002151enddef
2152
Bram Moolenaar2dd0a2c2020-08-08 15:10:27 +02002153def Test_cmd_modifier()
2154 tab echo '0'
Bram Moolenaard2c61702020-09-06 15:58:36 +02002155 CheckDefFailure(['5tab echo 3'], 'E16:')
Bram Moolenaar2dd0a2c2020-08-08 15:10:27 +02002156enddef
2157
2158def Test_restore_modifiers()
2159 # check that when compiling a :def function command modifiers are not messed
2160 # up.
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002161 var lines =<< trim END
Bram Moolenaar2dd0a2c2020-08-08 15:10:27 +02002162 vim9script
2163 set eventignore=
2164 autocmd QuickFixCmdPost * copen
2165 def AutocmdsDisabled()
Bram Moolenaar6cf7e3b2020-10-28 14:31:16 +01002166 eval 0
Bram Moolenaar2dd0a2c2020-08-08 15:10:27 +02002167 enddef
2168 func Func()
2169 noautocmd call s:AutocmdsDisabled()
2170 let g:ei_after = &eventignore
2171 endfunc
2172 Func()
2173 END
2174 CheckScriptSuccess(lines)
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002175 g:ei_after->assert_equal('')
Bram Moolenaar2dd0a2c2020-08-08 15:10:27 +02002176enddef
2177
Bram Moolenaardfa3d552020-09-10 22:05:08 +02002178def StackTop()
2179 eval 1
2180 eval 2
2181 # call not on fourth line
2182 StackBot()
2183enddef
2184
2185def StackBot()
2186 # throw an error
2187 eval [][0]
2188enddef
2189
2190def Test_callstack_def()
2191 try
2192 StackTop()
2193 catch
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002194 v:throwpoint->assert_match('Test_callstack_def\[2\]..StackTop\[4\]..StackBot, line 2')
Bram Moolenaardfa3d552020-09-10 22:05:08 +02002195 endtry
2196enddef
2197
Bram Moolenaare8211a32020-10-09 22:04:29 +02002198" Re-using spot for variable used in block
2199def Test_block_scoped_var()
2200 var lines =<< trim END
2201 vim9script
2202 def Func()
2203 var x = ['a', 'b', 'c']
2204 if 1
2205 var y = 'x'
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01002206 map(x, () => y)
Bram Moolenaare8211a32020-10-09 22:04:29 +02002207 endif
2208 var z = x
2209 assert_equal(['x', 'x', 'x'], z)
2210 enddef
2211 Func()
2212 END
2213 CheckScriptSuccess(lines)
2214enddef
2215
Bram Moolenaareeece9e2020-11-20 19:26:48 +01002216def Test_reset_did_emsg()
2217 var lines =<< trim END
2218 @s = 'blah'
2219 au BufWinLeave * #
2220 def Func()
2221 var winid = popup_create('popup', {})
2222 exe '*s'
2223 popup_close(winid)
2224 enddef
2225 Func()
2226 END
2227 CheckScriptFailure(lines, 'E492:', 8)
Bram Moolenaar2d870f82020-12-05 13:41:01 +01002228 delfunc! g:Func
Bram Moolenaareeece9e2020-11-20 19:26:48 +01002229enddef
2230
Bram Moolenaar57f799e2020-12-12 20:42:19 +01002231def Test_did_emsg_reset()
2232 # executing an autocommand resets did_emsg, this should not result in a
2233 # builtin function considered failing
2234 var lines =<< trim END
2235 vim9script
2236 au BufWinLeave * #
2237 def Func()
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01002238 popup_menu('', {callback: () => popup_create('', {})->popup_close()})
Bram Moolenaar57f799e2020-12-12 20:42:19 +01002239 eval [][0]
2240 enddef
2241 nno <F3> <cmd>call <sid>Func()<cr>
2242 feedkeys("\<F3>\e", 'xt')
2243 END
2244 writefile(lines, 'XemsgReset')
2245 assert_fails('so XemsgReset', ['E684:', 'E684:'], lines, 2)
2246 delete('XemsgReset')
2247 nunmap <F3>
2248 au! BufWinLeave
2249enddef
2250
Bram Moolenaar56602ba2020-12-05 21:22:08 +01002251def Test_abort_with_silent_call()
2252 var lines =<< trim END
2253 vim9script
2254 g:result = 'none'
2255 def Func()
2256 g:result += 3
2257 g:result = 'yes'
2258 enddef
2259 # error is silenced, but function aborts on error
2260 silent! Func()
2261 assert_equal('none', g:result)
2262 unlet g:result
2263 END
2264 CheckScriptSuccess(lines)
2265enddef
2266
Bram Moolenaarf665e972020-12-05 19:17:16 +01002267def Test_continues_with_silent_error()
2268 var lines =<< trim END
2269 vim9script
2270 g:result = 'none'
2271 def Func()
2272 silent! g:result += 3
2273 g:result = 'yes'
2274 enddef
2275 # error is silenced, function does not abort
2276 Func()
2277 assert_equal('yes', g:result)
2278 unlet g:result
2279 END
2280 CheckScriptSuccess(lines)
2281enddef
2282
Bram Moolenaaraf0df472020-12-02 20:51:22 +01002283def Test_abort_even_with_silent()
2284 var lines =<< trim END
2285 vim9script
2286 g:result = 'none'
2287 def Func()
2288 eval {-> ''}() .. '' .. {}['X']
2289 g:result = 'yes'
2290 enddef
Bram Moolenaarf665e972020-12-05 19:17:16 +01002291 silent! Func()
Bram Moolenaaraf0df472020-12-02 20:51:22 +01002292 assert_equal('none', g:result)
Bram Moolenaar4029cab2020-12-05 18:13:27 +01002293 unlet g:result
2294 END
2295 CheckScriptSuccess(lines)
2296enddef
2297
Bram Moolenaarf665e972020-12-05 19:17:16 +01002298def Test_cmdmod_silent_restored()
2299 var lines =<< trim END
2300 vim9script
2301 def Func()
2302 g:result = 'none'
2303 silent! g:result += 3
2304 g:result = 'none'
2305 g:result += 3
2306 enddef
2307 Func()
2308 END
2309 # can't use CheckScriptFailure, it ignores the :silent!
2310 var fname = 'Xdefsilent'
2311 writefile(lines, fname)
2312 var caught = 'no'
2313 try
2314 exe 'source ' .. fname
2315 catch /E1030:/
2316 caught = 'yes'
2317 assert_match('Func, line 4', v:throwpoint)
2318 endtry
2319 assert_equal('yes', caught)
2320 delete(fname)
2321enddef
2322
Bram Moolenaar4029cab2020-12-05 18:13:27 +01002323def Test_dict_member_with_silent()
2324 var lines =<< trim END
2325 vim9script
2326 g:result = 'none'
2327 var d: dict<any>
2328 def Func()
2329 try
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01002330 g:result = map([], (_, v) => ({}[v]))->join() .. d['']
Bram Moolenaar4029cab2020-12-05 18:13:27 +01002331 catch
2332 endtry
2333 enddef
2334 silent! Func()
2335 assert_equal('0', g:result)
2336 unlet g:result
Bram Moolenaaraf0df472020-12-02 20:51:22 +01002337 END
2338 CheckScriptSuccess(lines)
2339enddef
2340
Bram Moolenaarf9041332021-01-21 19:41:16 +01002341def Test_skip_cmds_with_silent()
2342 var lines =<< trim END
2343 vim9script
2344
2345 def Func(b: bool)
2346 Crash()
2347 enddef
2348
2349 def Crash()
2350 sil! :/not found/d _
2351 sil! :/not found/put _
2352 enddef
2353
2354 Func(true)
2355 END
2356 CheckScriptSuccess(lines)
2357enddef
2358
Bram Moolenaar5b3d1bb2020-12-22 12:20:08 +01002359def Test_opfunc()
2360 nnoremap <F3> <cmd>set opfunc=Opfunc<cr>g@
2361 def g:Opfunc(_: any): string
2362 setline(1, 'ASDF')
2363 return ''
2364 enddef
2365 new
2366 setline(1, 'asdf')
2367 feedkeys("\<F3>$", 'x')
2368 assert_equal('ASDF', getline(1))
2369
2370 bwipe!
2371 nunmap <F3>
2372enddef
2373
Bram Moolenaar077a4232020-12-22 18:33:27 +01002374" this was crashing on exit
2375def Test_nested_lambda_in_closure()
2376 var lines =<< trim END
2377 vim9script
2378 def Outer()
2379 def g:Inner()
2380 echo map([1, 2, 3], {_, v -> v + 1})
2381 enddef
2382 g:Inner()
2383 enddef
2384 defcompile
2385 writefile(['Done'], 'XnestedDone')
2386 quit
2387 END
2388 if !RunVim([], lines, '--clean')
2389 return
2390 endif
2391 assert_equal(['Done'], readfile('XnestedDone'))
2392 delete('XnestedDone')
2393enddef
2394
Bram Moolenaar04947cc2021-03-06 19:26:46 +01002395def Test_check_func_arg_types()
2396 var lines =<< trim END
2397 vim9script
2398 def F1(x: string): string
2399 return x
2400 enddef
2401
2402 def F2(x: number): number
2403 return x + 1
2404 enddef
2405
2406 def G(g: func): dict<func>
2407 return {f: g}
2408 enddef
2409
2410 def H(d: dict<func>): string
2411 return d.f('a')
2412 enddef
2413 END
2414
2415 CheckScriptSuccess(lines + ['echo H(G(F1))'])
2416 CheckScriptFailure(lines + ['echo H(G(F2))'], 'E1013:')
2417enddef
2418
Bram Moolenaar077a4232020-12-22 18:33:27 +01002419
Bram Moolenaarf7779c62020-05-03 15:38:16 +02002420
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002421" vim: ts=8 sw=2 sts=2 expandtab tw=80 fdm=marker