blob: cd2a15d2da9b55b9c9ffe3c644d48584d344bd4a [file] [log] [blame]
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001" Test various aspects of the Vim9 script language.
2
3source check.vim
Bram Moolenaarad304702020-09-06 18:22:53 +02004source term_util.vim
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02005source view_util.vim
Bram Moolenaar04b12692020-05-04 23:24:44 +02006source vim9.vim
Bram Moolenaar47e7d702020-07-05 18:18:42 +02007source screendump.vim
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02008
9func Test_def_basic()
10 def SomeFunc(): string
11 return 'yes'
12 enddef
Bram Moolenaarc0c71e92020-09-11 19:09:48 +020013 call SomeFunc()->assert_equal('yes')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +020014endfunc
15
Bram Moolenaar2b9b17e2020-10-13 18:38:11 +020016func Test_compiling_error()
17 " use a terminal to see the whole error message
Bram Moolenaarf4e8cdd2020-10-12 22:07:13 +020018 CheckRunVimInTerminal
19
Bram Moolenaar2b9b17e2020-10-13 18:38:11 +020020 call TestCompilingError()
Bram Moolenaare8c46602021-04-05 22:27:37 +020021 call TestCompilingErrorInTry()
Bram Moolenaar2b9b17e2020-10-13 18:38:11 +020022endfunc
23
24def TestCompilingError()
Bram Moolenaarf4e8cdd2020-10-12 22:07:13 +020025 var lines =<< trim END
26 vim9script
27 def Fails()
28 echo nothing
29 enddef
30 defcompile
31 END
Bram Moolenaare8c46602021-04-05 22:27:37 +020032 writefile(lines, 'XTest_compile_error')
Bram Moolenaarf4e8cdd2020-10-12 22:07:13 +020033 var buf = RunVimInTerminal('-S XTest_compile_error',
Bram Moolenaare0de1712020-12-02 17:36:54 +010034 {rows: 10, wait_for_ruler: 0})
Bram Moolenaare8c46602021-04-05 22:27:37 +020035 WaitForAssert(() => assert_match('Error detected while compiling command line.*Fails.*Variable not found: nothing',
Bram Moolenaar03dfde22021-02-14 13:17:22 +010036 Term_getlines(buf, range(1, 9))))
Bram Moolenaarf4e8cdd2020-10-12 22:07:13 +020037
38 # clean up
Bram Moolenaare8c46602021-04-05 22:27:37 +020039 StopVimInTerminal(buf)
40 delete('XTest_compile_error')
41enddef
42
43def TestCompilingErrorInTry()
44 var dir = 'Xdir/autoload'
45 mkdir(dir, 'p')
46
47 var lines =<< trim END
48 vim9script
49 def script#OnlyCompiled()
50 g:runtime = 'yes'
51 invalid
52 enddef
53 END
54 writefile(lines, dir .. '/script.vim')
55
56 lines =<< trim END
57 vim9script
58 todo
59 try
60 script#OnlyCompiled()
61 catch /nothing/
62 endtry
63 END
64 lines[1] = 'set rtp=' .. getcwd() .. '/Xdir'
65 writefile(lines, 'XTest_compile_error')
66
67 var buf = RunVimInTerminal('-S XTest_compile_error', {rows: 10, wait_for_ruler: 0})
68 WaitForAssert(() => assert_match('Error detected while compiling command line.*function script#OnlyCompiled.*Invalid command: invalid',
69 Term_getlines(buf, range(1, 9))))
70
71 # clean up
72 StopVimInTerminal(buf)
73 delete('XTest_compile_error')
74 delete('Xdir', 'rf')
Bram Moolenaarf4e8cdd2020-10-12 22:07:13 +020075enddef
76
Bram Moolenaarb55d6182021-06-08 22:01:53 +020077def Test_compile_error_in_called_function()
78 var lines =<< trim END
79 vim9script
80 var n: number
81 def Foo()
82 &hls = n
83 enddef
84 def Bar()
85 Foo()
86 enddef
87 silent! Foo()
88 Bar()
89 END
90 CheckScriptFailureList(lines, ['E1012:', 'E1191:'])
91enddef
92
Bram Moolenaarf48b2fa2021-04-12 22:02:36 +020093def Test_autoload_name_mismatch()
94 var dir = 'Xdir/autoload'
95 mkdir(dir, 'p')
96
97 var lines =<< trim END
98 vim9script
99 def scriptX#Function()
100 # comment
101 g:runtime = 'yes'
102 enddef
103 END
104 writefile(lines, dir .. '/script.vim')
105
106 var save_rtp = &rtp
107 exe 'set rtp=' .. getcwd() .. '/Xdir'
108 lines =<< trim END
109 call script#Function()
110 END
111 CheckScriptFailure(lines, 'E746:', 2)
112
113 &rtp = save_rtp
114 delete(dir, 'rf')
115enddef
116
Bram Moolenaar0ba48e82020-11-17 18:23:19 +0100117def CallRecursive(n: number): number
118 return CallRecursive(n + 1)
119enddef
120
121def CallMapRecursive(l: list<number>): number
Bram Moolenaar2949cfd2020-12-31 21:28:47 +0100122 return map(l, (_, v) => CallMapRecursive([v]))[0]
Bram Moolenaar0ba48e82020-11-17 18:23:19 +0100123enddef
124
125def Test_funcdepth_error()
126 set maxfuncdepth=10
127
128 var caught = false
129 try
130 CallRecursive(1)
131 catch /E132:/
132 caught = true
133 endtry
134 assert_true(caught)
135
136 caught = false
137 try
138 CallMapRecursive([1])
139 catch /E132:/
140 caught = true
141 endtry
142 assert_true(caught)
143
144 set maxfuncdepth&
145enddef
146
Bram Moolenaar5178b1b2021-01-01 18:43:51 +0100147def Test_endfunc_enddef()
148 var lines =<< trim END
149 def Test()
150 echo 'test'
151 endfunc
152 enddef
153 END
154 CheckScriptFailure(lines, 'E1151:', 3)
155
156 lines =<< trim END
157 def Test()
158 func Nested()
159 echo 'test'
160 enddef
161 enddef
162 END
163 CheckScriptFailure(lines, 'E1152:', 4)
Bram Moolenaar49f1e9e2021-03-22 20:49:02 +0100164
165 lines =<< trim END
166 def Ok()
167 echo 'hello'
168 enddef | echo 'there'
169 def Bad()
170 echo 'hello'
171 enddef there
172 END
173 CheckScriptFailure(lines, 'E1173: Text found after enddef: there', 6)
Bram Moolenaar5178b1b2021-01-01 18:43:51 +0100174enddef
175
Bram Moolenaarb8ba9b92021-01-01 18:54:34 +0100176def Test_missing_endfunc_enddef()
177 var lines =<< trim END
178 vim9script
179 def Test()
180 echo 'test'
181 endef
182 END
183 CheckScriptFailure(lines, 'E1057:', 2)
184
185 lines =<< trim END
186 vim9script
187 func Some()
188 echo 'test'
189 enfffunc
190 END
191 CheckScriptFailure(lines, 'E126:', 2)
192enddef
193
Bram Moolenaar4efd9942021-01-24 21:14:20 +0100194def Test_white_space_before_paren()
195 var lines =<< trim END
196 vim9script
197 def Test ()
198 echo 'test'
199 enddef
200 END
201 CheckScriptFailure(lines, 'E1068:', 2)
202
203 lines =<< trim END
204 vim9script
205 func Test ()
206 echo 'test'
207 endfunc
208 END
209 CheckScriptFailure(lines, 'E1068:', 2)
210
211 lines =<< trim END
212 def Test ()
213 echo 'test'
214 enddef
215 END
216 CheckScriptFailure(lines, 'E1068:', 1)
217
218 lines =<< trim END
219 func Test ()
220 echo 'test'
221 endfunc
222 END
223 CheckScriptSuccess(lines)
224enddef
225
Bram Moolenaar832ea892021-01-08 21:55:26 +0100226def Test_enddef_dict_key()
227 var d = {
228 enddef: 'x',
229 endfunc: 'y',
230 }
231 assert_equal({enddef: 'x', endfunc: 'y'}, d)
232enddef
233
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200234def ReturnString(): string
235 return 'string'
236enddef
237
238def ReturnNumber(): number
239 return 123
240enddef
241
242let g:notNumber = 'string'
243
244def ReturnGlobal(): number
245 return g:notNumber
246enddef
247
248def Test_return_something()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +0200249 ReturnString()->assert_equal('string')
250 ReturnNumber()->assert_equal(123)
Bram Moolenaar5e654232020-09-16 15:22:00 +0200251 assert_fails('ReturnGlobal()', 'E1012: Type mismatch; expected number but got string', '', 1, 'ReturnGlobal')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200252enddef
253
Bram Moolenaare32e5162021-01-21 20:21:29 +0100254def Test_check_argument_type()
255 var lines =<< trim END
256 vim9script
257 def Val(a: number, b: number): number
258 return 0
259 enddef
260 def Func()
261 var x: any = true
262 Val(0, x)
263 enddef
264 disass Func
265 Func()
266 END
267 CheckScriptFailure(lines, 'E1013: Argument 2: type mismatch, expected number but got bool', 2)
268enddef
269
Bram Moolenaarefd88552020-06-18 20:50:10 +0200270def Test_missing_return()
271 CheckDefFailure(['def Missing(): number',
272 ' if g:cond',
273 ' echo "no return"',
274 ' else',
275 ' return 0',
276 ' endif'
277 'enddef'], 'E1027:')
278 CheckDefFailure(['def Missing(): number',
279 ' if g:cond',
280 ' return 1',
281 ' else',
282 ' echo "no return"',
283 ' endif'
284 'enddef'], 'E1027:')
285 CheckDefFailure(['def Missing(): number',
286 ' if g:cond',
287 ' return 1',
288 ' else',
289 ' return 2',
290 ' endif'
291 ' return 3'
292 'enddef'], 'E1095:')
293enddef
294
Bram Moolenaar403dc312020-10-17 19:29:51 +0200295def Test_return_bool()
296 var lines =<< trim END
297 vim9script
298 def MenuFilter(id: number, key: string): bool
299 return popup_filter_menu(id, key)
300 enddef
301 def YesnoFilter(id: number, key: string): bool
302 return popup_filter_yesno(id, key)
303 enddef
304 defcompile
305 END
306 CheckScriptSuccess(lines)
307enddef
308
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200309let s:nothing = 0
310def ReturnNothing()
311 s:nothing = 1
312 if true
313 return
314 endif
315 s:nothing = 2
316enddef
317
318def Test_return_nothing()
319 ReturnNothing()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +0200320 s:nothing->assert_equal(1)
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200321enddef
322
Bram Moolenaar648ea762021-01-15 19:04:32 +0100323def Test_return_invalid()
324 var lines =<< trim END
325 vim9script
326 def Func(): invalid
327 return xxx
328 enddef
329 defcompile
330 END
331 CheckScriptFailure(lines, 'E1010:', 2)
Bram Moolenaar31842cd2021-02-12 22:10:21 +0100332
333 lines =<< trim END
334 vim9script
335 def Test(Fun: func(number): number): list<number>
336 return map([1, 2, 3], (_, i) => Fun(i))
337 enddef
338 defcompile
339 def Inc(nr: number): nr
340 return nr + 2
341 enddef
342 echo Test(Inc)
343 END
344 # doing this twice was leaking memory
345 CheckScriptFailure(lines, 'E1010:')
346 CheckScriptFailure(lines, 'E1010:')
Bram Moolenaar648ea762021-01-15 19:04:32 +0100347enddef
348
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200349func Increment()
350 let g:counter += 1
351endfunc
352
353def Test_call_ufunc_count()
354 g:counter = 1
355 Increment()
356 Increment()
357 Increment()
Bram Moolenaarf5be8cd2020-07-17 20:36:00 +0200358 # works with and without :call
Bram Moolenaarc0c71e92020-09-11 19:09:48 +0200359 g:counter->assert_equal(4)
360 eval g:counter->assert_equal(4)
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200361 unlet g:counter
362enddef
363
364def MyVarargs(arg: string, ...rest: list<string>): string
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +0200365 var res = arg
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200366 for s in rest
367 res ..= ',' .. s
368 endfor
369 return res
370enddef
371
372def Test_call_varargs()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +0200373 MyVarargs('one')->assert_equal('one')
374 MyVarargs('one', 'two')->assert_equal('one,two')
375 MyVarargs('one', 'two', 'three')->assert_equal('one,two,three')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200376enddef
377
378def MyDefaultArgs(name = 'string'): string
379 return name
380enddef
381
Bram Moolenaare30f64b2020-07-15 19:48:20 +0200382def MyDefaultSecond(name: string, second: bool = true): string
383 return second ? name : 'none'
384enddef
385
Bram Moolenaar38a3bfa2021-03-29 22:14:55 +0200386
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200387def Test_call_default_args()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +0200388 MyDefaultArgs()->assert_equal('string')
Bram Moolenaar38a3bfa2021-03-29 22:14:55 +0200389 MyDefaultArgs(v:none)->assert_equal('string')
Bram Moolenaarc0c71e92020-09-11 19:09:48 +0200390 MyDefaultArgs('one')->assert_equal('one')
Bram Moolenaar38a3bfa2021-03-29 22:14:55 +0200391 assert_fails('MyDefaultArgs("one", "two")', 'E118:', '', 4, 'Test_call_default_args')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200392
Bram Moolenaarc0c71e92020-09-11 19:09:48 +0200393 MyDefaultSecond('test')->assert_equal('test')
394 MyDefaultSecond('test', true)->assert_equal('test')
395 MyDefaultSecond('test', false)->assert_equal('none')
Bram Moolenaare30f64b2020-07-15 19:48:20 +0200396
Bram Moolenaar38a3bfa2021-03-29 22:14:55 +0200397 var lines =<< trim END
398 def MyDefaultThird(name: string, aa = 'aa', bb = 'bb'): string
399 return name .. aa .. bb
400 enddef
401
402 MyDefaultThird('->')->assert_equal('->aabb')
403 MyDefaultThird('->', v:none)->assert_equal('->aabb')
404 MyDefaultThird('->', 'xx')->assert_equal('->xxbb')
405 MyDefaultThird('->', v:none, v:none)->assert_equal('->aabb')
406 MyDefaultThird('->', 'xx', v:none)->assert_equal('->xxbb')
407 MyDefaultThird('->', v:none, 'yy')->assert_equal('->aayy')
408 MyDefaultThird('->', 'xx', 'yy')->assert_equal('->xxyy')
409 END
410 CheckDefAndScriptSuccess(lines)
411
Bram Moolenaar822ba242020-05-24 23:00:18 +0200412 CheckScriptFailure(['def Func(arg: number = asdf)', 'enddef', 'defcompile'], 'E1001:')
Bram Moolenaar2d870f82020-12-05 13:41:01 +0100413 delfunc g:Func
Bram Moolenaar77072282020-09-16 17:55:40 +0200414 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 +0100415 delfunc g:Func
Yegappan Lakshmanan34fcb692021-05-25 20:14:00 +0200416 CheckDefFailure(['def Func(x: number = )', 'enddef'], 'E15:')
Bram Moolenaar12bce952021-03-11 20:04:04 +0100417
Bram Moolenaar38a3bfa2021-03-29 22:14:55 +0200418 lines =<< trim END
Bram Moolenaar12bce952021-03-11 20:04:04 +0100419 vim9script
420 def Func(a = b == 0 ? 1 : 2, b = 0)
421 enddef
422 defcompile
423 END
424 CheckScriptFailure(lines, 'E1001: Variable not found: b')
Bram Moolenaar04b12692020-05-04 23:24:44 +0200425enddef
426
Bram Moolenaarcef12702021-01-04 14:09:43 +0100427def FuncWithComment( # comment
428 a: number, #comment
429 b: bool, # comment
430 c: string) #comment
431 assert_equal(4, a)
432 assert_equal(true, b)
433 assert_equal('yes', c)
434enddef
435
436def Test_func_with_comments()
437 FuncWithComment(4, true, 'yes')
438
439 var lines =<< trim END
440 def Func(# comment
441 arg: string)
442 enddef
443 END
444 CheckScriptFailure(lines, 'E125:', 1)
445
446 lines =<< trim END
447 def Func(
448 arg: string# comment
449 )
450 enddef
451 END
452 CheckScriptFailure(lines, 'E475:', 2)
453
454 lines =<< trim END
455 def Func(
456 arg: string
457 )# comment
458 enddef
459 END
460 CheckScriptFailure(lines, 'E488:', 3)
461enddef
462
Bram Moolenaar04b12692020-05-04 23:24:44 +0200463def Test_nested_function()
464 def Nested(arg: string): string
465 return 'nested ' .. arg
466 enddef
Bram Moolenaarc0c71e92020-09-11 19:09:48 +0200467 Nested('function')->assert_equal('nested function')
Bram Moolenaar04b12692020-05-04 23:24:44 +0200468
Bram Moolenaar0e65d3d2020-05-05 17:53:16 +0200469 CheckDefFailure(['def Nested()', 'enddef', 'Nested(66)'], 'E118:')
470 CheckDefFailure(['def Nested(arg: string)', 'enddef', 'Nested()'], 'E119:')
471
Bram Moolenaar04b12692020-05-04 23:24:44 +0200472 CheckDefFailure(['func Nested()', 'endfunc'], 'E1086:')
Bram Moolenaarbcbf4132020-08-01 22:35:13 +0200473 CheckDefFailure(['def s:Nested()', 'enddef'], 'E1075:')
474 CheckDefFailure(['def b:Nested()', 'enddef'], 'E1075:')
Bram Moolenaar8b848ca2020-09-10 22:28:01 +0200475
Bram Moolenaar54021752020-12-06 18:50:36 +0100476 var lines =<< trim END
477 def Outer()
478 def Inner()
479 # comment
480 enddef
481 def Inner()
482 enddef
483 enddef
484 END
485 CheckDefFailure(lines, 'E1073:')
486
487 lines =<< trim END
488 def Outer()
489 def Inner()
490 # comment
491 enddef
492 def! Inner()
493 enddef
494 enddef
495 END
496 CheckDefFailure(lines, 'E1117:')
497
498 # nested function inside conditional
Bram Moolenaar54021752020-12-06 18:50:36 +0100499 lines =<< trim END
500 vim9script
501 var thecount = 0
502 if true
503 def Test(): number
504 def TheFunc(): number
505 thecount += 1
506 return thecount
507 enddef
508 return TheFunc()
509 enddef
510 endif
511 defcompile
512 assert_equal(1, Test())
513 assert_equal(2, Test())
514 END
515 CheckScriptSuccess(lines)
Bram Moolenaar8863bda2021-03-17 18:42:08 +0100516
517 # also works when "thecount" is inside the "if" block
518 lines =<< trim END
519 vim9script
520 if true
521 var thecount = 0
522 def Test(): number
523 def TheFunc(): number
524 thecount += 1
525 return thecount
526 enddef
527 return TheFunc()
528 enddef
529 endif
530 defcompile
531 assert_equal(1, Test())
532 assert_equal(2, Test())
533 END
534 CheckScriptSuccess(lines)
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200535enddef
536
Bram Moolenaaradc8e442020-12-31 18:28:18 +0100537def Test_not_nested_function()
538 echo printf('%d',
539 function('len')('xxx'))
540enddef
541
Bram Moolenaaraf8edbb2020-08-01 00:03:09 +0200542func Test_call_default_args_from_func()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +0200543 call MyDefaultArgs()->assert_equal('string')
544 call MyDefaultArgs('one')->assert_equal('one')
Bram Moolenaar9bd5d872020-09-06 21:47:48 +0200545 call assert_fails('call MyDefaultArgs("one", "two")', 'E118:', '', 3, 'Test_call_default_args_from_func')
Bram Moolenaaraf8edbb2020-08-01 00:03:09 +0200546endfunc
547
Bram Moolenaar38ddf332020-07-31 22:05:04 +0200548def Test_nested_global_function()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +0200549 var lines =<< trim END
Bram Moolenaar38ddf332020-07-31 22:05:04 +0200550 vim9script
551 def Outer()
552 def g:Inner(): string
553 return 'inner'
554 enddef
555 enddef
Bram Moolenaaraf8edbb2020-08-01 00:03:09 +0200556 defcompile
557 Outer()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +0200558 g:Inner()->assert_equal('inner')
Bram Moolenaaraf8edbb2020-08-01 00:03:09 +0200559 delfunc g:Inner
560 Outer()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +0200561 g:Inner()->assert_equal('inner')
Bram Moolenaaraf8edbb2020-08-01 00:03:09 +0200562 delfunc g:Inner
563 Outer()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +0200564 g:Inner()->assert_equal('inner')
Bram Moolenaaraf8edbb2020-08-01 00:03:09 +0200565 delfunc g:Inner
Bram Moolenaar38ddf332020-07-31 22:05:04 +0200566 END
567 CheckScriptSuccess(lines)
Bram Moolenaar2c79e9d2020-08-01 18:57:52 +0200568
569 lines =<< trim END
570 vim9script
571 def Outer()
572 def g:Inner(): string
573 return 'inner'
574 enddef
575 enddef
576 defcompile
577 Outer()
578 Outer()
579 END
580 CheckScriptFailure(lines, "E122:")
Bram Moolenaarcd45ed02020-12-22 17:35:54 +0100581 delfunc g:Inner
Bram Moolenaarad486a02020-08-01 23:22:18 +0200582
583 lines =<< trim END
584 vim9script
Bram Moolenaar58a52f22020-12-22 18:56:55 +0100585 def Outer()
586 def g:Inner()
Bram Moolenaar2949cfd2020-12-31 21:28:47 +0100587 echo map([1, 2, 3], (_, v) => v + 1)
Bram Moolenaar58a52f22020-12-22 18:56:55 +0100588 enddef
589 g:Inner()
590 enddef
591 Outer()
592 END
593 CheckScriptSuccess(lines)
594 delfunc g:Inner
595
596 lines =<< trim END
597 vim9script
Bram Moolenaarad486a02020-08-01 23:22:18 +0200598 def Func()
599 echo 'script'
600 enddef
601 def Outer()
602 def Func()
603 echo 'inner'
604 enddef
605 enddef
606 defcompile
607 END
608 CheckScriptFailure(lines, "E1073:")
Bram Moolenaar38ddf332020-07-31 22:05:04 +0200609enddef
610
Bram Moolenaar6abdcf82020-11-22 18:15:44 +0100611def DefListAll()
612 def
613enddef
614
615def DefListOne()
616 def DefListOne
617enddef
618
619def DefListMatches()
620 def /DefList
621enddef
622
623def Test_nested_def_list()
624 var funcs = split(execute('call DefListAll()'), "\n")
625 assert_true(len(funcs) > 10)
626 assert_true(funcs->index('def DefListAll()') >= 0)
627
628 funcs = split(execute('call DefListOne()'), "\n")
629 assert_equal([' def DefListOne()', '1 def DefListOne', ' enddef'], funcs)
630
631 funcs = split(execute('call DefListMatches()'), "\n")
632 assert_true(len(funcs) >= 3)
633 assert_true(funcs->index('def DefListAll()') >= 0)
634 assert_true(funcs->index('def DefListOne()') >= 0)
635 assert_true(funcs->index('def DefListMatches()') >= 0)
Bram Moolenaar54021752020-12-06 18:50:36 +0100636
637 var lines =<< trim END
638 vim9script
639 def Func()
640 def +Func+
641 enddef
642 defcompile
643 END
644 CheckScriptFailure(lines, 'E476:', 1)
Bram Moolenaar6abdcf82020-11-22 18:15:44 +0100645enddef
646
Bram Moolenaar333894b2020-08-01 18:53:07 +0200647def Test_global_local_function()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +0200648 var lines =<< trim END
Bram Moolenaar333894b2020-08-01 18:53:07 +0200649 vim9script
650 def g:Func(): string
651 return 'global'
652 enddef
653 def Func(): string
654 return 'local'
655 enddef
Bram Moolenaarc0c71e92020-09-11 19:09:48 +0200656 g:Func()->assert_equal('global')
657 Func()->assert_equal('local')
Bram Moolenaar2d870f82020-12-05 13:41:01 +0100658 delfunc g:Func
Bram Moolenaar333894b2020-08-01 18:53:07 +0200659 END
660 CheckScriptSuccess(lines)
Bram Moolenaar035d6e92020-08-11 22:30:42 +0200661
662 lines =<< trim END
663 vim9script
664 def g:Funcy()
665 echo 'funcy'
666 enddef
667 s:Funcy()
668 END
669 CheckScriptFailure(lines, 'E117:')
Bram Moolenaar333894b2020-08-01 18:53:07 +0200670enddef
671
Bram Moolenaar0f769812020-09-12 18:32:34 +0200672def Test_local_function_shadows_global()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +0200673 var lines =<< trim END
Bram Moolenaar0f769812020-09-12 18:32:34 +0200674 vim9script
675 def g:Gfunc(): string
676 return 'global'
677 enddef
678 def AnotherFunc(): number
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +0200679 var Gfunc = function('len')
Bram Moolenaar0f769812020-09-12 18:32:34 +0200680 return Gfunc('testing')
681 enddef
682 g:Gfunc()->assert_equal('global')
683 AnotherFunc()->assert_equal(7)
684 delfunc g:Gfunc
685 END
686 CheckScriptSuccess(lines)
687
688 lines =<< trim END
689 vim9script
690 def g:Func(): string
691 return 'global'
692 enddef
693 def AnotherFunc()
694 g:Func = function('len')
695 enddef
696 AnotherFunc()
697 END
698 CheckScriptFailure(lines, 'E705:')
699 delfunc g:Func
Bram Moolenaar0865b152021-04-05 15:38:51 +0200700
701 # global function is found without g: prefix
702 lines =<< trim END
703 vim9script
704 def g:Func(): string
705 return 'global'
706 enddef
707 def AnotherFunc(): string
708 return Func()
709 enddef
710 assert_equal('global', AnotherFunc())
711 delfunc g:Func
712 END
713 CheckScriptSuccess(lines)
714
715 lines =<< trim END
716 vim9script
717 def g:Func(): string
718 return 'global'
719 enddef
720 assert_equal('global', Func())
721 delfunc g:Func
722 END
723 CheckScriptSuccess(lines)
Bram Moolenaar0f769812020-09-12 18:32:34 +0200724enddef
725
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200726func TakesOneArg(arg)
727 echo a:arg
728endfunc
729
730def Test_call_wrong_args()
Bram Moolenaard2c61702020-09-06 15:58:36 +0200731 CheckDefFailure(['TakesOneArg()'], 'E119:')
732 CheckDefFailure(['TakesOneArg(11, 22)'], 'E118:')
733 CheckDefFailure(['bufnr(xxx)'], 'E1001:')
734 CheckScriptFailure(['def Func(Ref: func(s: string))'], 'E475:')
Bram Moolenaaree8580e2020-08-28 17:19:07 +0200735
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +0200736 var lines =<< trim END
Bram Moolenaaree8580e2020-08-28 17:19:07 +0200737 vim9script
738 def Func(s: string)
739 echo s
740 enddef
741 Func([])
742 END
Bram Moolenaar77072282020-09-16 17:55:40 +0200743 CheckScriptFailure(lines, 'E1013: Argument 1: type mismatch, expected string but got list<unknown>', 5)
Bram Moolenaarb185a402020-09-18 22:42:00 +0200744
745 lines =<< trim END
746 vim9script
Bram Moolenaarb4893b82021-02-21 22:20:24 +0100747 var name = 'piet'
748 def FuncOne(name: string)
749 echo nr
750 enddef
751 END
Bram Moolenaar057e84a2021-02-28 16:55:11 +0100752 CheckScriptFailure(lines, 'E1168:')
Bram Moolenaarb4893b82021-02-21 22:20:24 +0100753
754 lines =<< trim END
755 vim9script
Bram Moolenaarb185a402020-09-18 22:42:00 +0200756 def FuncOne(nr: number)
757 echo nr
758 enddef
759 def FuncTwo()
760 FuncOne()
761 enddef
762 defcompile
763 END
764 writefile(lines, 'Xscript')
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +0200765 var didCatch = false
Bram Moolenaarb185a402020-09-18 22:42:00 +0200766 try
767 source Xscript
768 catch
769 assert_match('E119: Not enough arguments for function: <SNR>\d\+_FuncOne', v:exception)
770 assert_match('Xscript\[8\]..function <SNR>\d\+_FuncTwo, line 1', v:throwpoint)
771 didCatch = true
772 endtry
773 assert_true(didCatch)
774
775 lines =<< trim END
776 vim9script
777 def FuncOne(nr: number)
778 echo nr
779 enddef
780 def FuncTwo()
781 FuncOne(1, 2)
782 enddef
783 defcompile
784 END
785 writefile(lines, 'Xscript')
786 didCatch = false
787 try
788 source Xscript
789 catch
790 assert_match('E118: Too many arguments for function: <SNR>\d\+_FuncOne', v:exception)
791 assert_match('Xscript\[8\]..function <SNR>\d\+_FuncTwo, line 1', v:throwpoint)
792 didCatch = true
793 endtry
794 assert_true(didCatch)
795
796 delete('Xscript')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200797enddef
798
Bram Moolenaar50824712020-12-20 21:10:17 +0100799def Test_call_funcref_wrong_args()
800 var head =<< trim END
801 vim9script
802 def Func3(a1: string, a2: number, a3: list<number>)
803 echo a1 .. a2 .. a3[0]
804 enddef
805 def Testme()
806 var funcMap: dict<func> = {func: Func3}
807 END
808 var tail =<< trim END
809 enddef
810 Testme()
811 END
812 CheckScriptSuccess(head + ["funcMap['func']('str', 123, [1, 2, 3])"] + tail)
813
814 CheckScriptFailure(head + ["funcMap['func']('str', 123)"] + tail, 'E119:')
815 CheckScriptFailure(head + ["funcMap['func']('str', 123, [1], 4)"] + tail, 'E118:')
Bram Moolenaar32b3f822021-01-06 21:59:39 +0100816
817 var lines =<< trim END
818 vim9script
819 var Ref: func(number): any
820 Ref = (j) => !j
821 echo Ref(false)
822 END
823 CheckScriptFailure(lines, 'E1013: Argument 1: type mismatch, expected number but got bool', 4)
824
825 lines =<< trim END
826 vim9script
827 var Ref: func(number): any
828 Ref = (j) => !j
829 call Ref(false)
830 END
831 CheckScriptFailure(lines, 'E1013: Argument 1: type mismatch, expected number but got bool', 4)
Bram Moolenaar50824712020-12-20 21:10:17 +0100832enddef
833
Bram Moolenaarb4d16cb2020-11-05 18:45:46 +0100834def Test_call_lambda_args()
Bram Moolenaar2a389082021-04-09 20:24:31 +0200835 var lines =<< trim END
836 var Callback = (..._) => 'anything'
837 assert_equal('anything', Callback())
838 assert_equal('anything', Callback(1))
839 assert_equal('anything', Callback('a', 2))
Bram Moolenaar1088b692021-04-09 22:12:44 +0200840
841 assert_equal('xyz', ((a: string): string => a)('xyz'))
Bram Moolenaar2a389082021-04-09 20:24:31 +0200842 END
843 CheckDefAndScriptSuccess(lines)
844
Bram Moolenaar2949cfd2020-12-31 21:28:47 +0100845 CheckDefFailure(['echo ((i) => 0)()'],
846 'E119: Not enough arguments for function: ((i) => 0)()')
Bram Moolenaarb4d16cb2020-11-05 18:45:46 +0100847
Bram Moolenaar2a389082021-04-09 20:24:31 +0200848 lines =<< trim END
Bram Moolenaar2949cfd2020-12-31 21:28:47 +0100849 var Ref = (x: number, y: number) => x + y
Bram Moolenaarb4d16cb2020-11-05 18:45:46 +0100850 echo Ref(1, 'x')
851 END
852 CheckDefFailure(lines, 'E1013: Argument 2: type mismatch, expected number but got string')
Bram Moolenaare68b02a2021-01-03 13:09:51 +0100853
854 lines =<< trim END
855 var Ref: func(job, string, number)
856 Ref = (x, y) => 0
857 END
858 CheckDefAndScriptFailure(lines, 'E1012:')
859
860 lines =<< trim END
861 var Ref: func(job, string)
862 Ref = (x, y, z) => 0
863 END
864 CheckDefAndScriptFailure(lines, 'E1012:')
Bram Moolenaar057e84a2021-02-28 16:55:11 +0100865
866 lines =<< trim END
867 var one = 1
868 var l = [1, 2, 3]
869 echo map(l, (one) => one)
870 END
871 CheckDefFailure(lines, 'E1167:')
872 CheckScriptFailure(['vim9script'] + lines, 'E1168:')
873
874 lines =<< trim END
875 def ShadowLocal()
876 var one = 1
877 var l = [1, 2, 3]
878 echo map(l, (one) => one)
879 enddef
880 END
881 CheckDefFailure(lines, 'E1167:')
882
883 lines =<< trim END
884 def Shadowarg(one: number)
885 var l = [1, 2, 3]
886 echo map(l, (one) => one)
887 enddef
888 END
889 CheckDefFailure(lines, 'E1167:')
Bram Moolenaar767034c2021-04-09 17:24:52 +0200890
891 lines =<< trim END
892 echo ((a) => a)('aa', 'bb')
893 END
894 CheckDefAndScriptFailure(lines, 'E118:', 1)
Bram Moolenaarb4d16cb2020-11-05 18:45:46 +0100895enddef
896
Bram Moolenaar5f91e742021-03-17 21:29:29 +0100897def FilterWithCond(x: string, Cond: func(string): bool): bool
898 return Cond(x)
899enddef
900
Bram Moolenaar0346b792021-01-31 22:18:29 +0100901def Test_lambda_return_type()
902 var lines =<< trim END
903 var Ref = (): => 123
904 END
905 CheckDefAndScriptFailure(lines, 'E1157:', 1)
Bram Moolenaar5f91e742021-03-17 21:29:29 +0100906
Yegappan Lakshmanan611728f2021-05-24 15:15:47 +0200907 # no space before the return type
908 lines =<< trim END
909 var Ref = (x):number => x + 1
910 END
911 CheckDefAndScriptFailure(lines, 'E1069:', 1)
912
Bram Moolenaar5f91e742021-03-17 21:29:29 +0100913 # this works
914 for x in ['foo', 'boo']
915 echo FilterWithCond(x, (v) => v =~ '^b')
916 endfor
917
918 # this fails
919 lines =<< trim END
920 echo FilterWithCond('foo', (v) => v .. '^b')
921 END
922 CheckDefAndScriptFailure(lines, 'E1013: Argument 2: type mismatch, expected func(string): bool but got func(any): string', 1)
Bram Moolenaar0346b792021-01-31 22:18:29 +0100923enddef
924
Bram Moolenaar709664c2020-12-12 14:33:41 +0100925def Test_lambda_uses_assigned_var()
926 CheckDefSuccess([
927 'var x: any = "aaa"'
Bram Moolenaar2949cfd2020-12-31 21:28:47 +0100928 'x = filter(["bbb"], (_, v) => v =~ x)'])
Bram Moolenaar709664c2020-12-12 14:33:41 +0100929enddef
930
Bram Moolenaar18062fc2021-03-05 21:35:47 +0100931def Test_pass_legacy_lambda_to_def_func()
932 var lines =<< trim END
933 vim9script
934 func Foo()
935 eval s:Bar({x -> 0})
936 endfunc
937 def Bar(y: any)
938 enddef
939 Foo()
940 END
941 CheckScriptSuccess(lines)
942enddef
943
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200944" Default arg and varargs
945def MyDefVarargs(one: string, two = 'foo', ...rest: list<string>): string
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +0200946 var res = one .. ',' .. two
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +0200947 for s in rest
948 res ..= ',' .. s
949 endfor
950 return res
951enddef
952
953def Test_call_def_varargs()
Bram Moolenaar9bd5d872020-09-06 21:47:48 +0200954 assert_fails('MyDefVarargs()', 'E119:', '', 1, 'Test_call_def_varargs')
Bram Moolenaarc0c71e92020-09-11 19:09:48 +0200955 MyDefVarargs('one')->assert_equal('one,foo')
956 MyDefVarargs('one', 'two')->assert_equal('one,two')
957 MyDefVarargs('one', 'two', 'three')->assert_equal('one,two,three')
Bram Moolenaar24aa48b2020-07-25 16:33:02 +0200958 CheckDefFailure(['MyDefVarargs("one", 22)'],
Bram Moolenaar77072282020-09-16 17:55:40 +0200959 'E1013: Argument 2: type mismatch, expected string but got number')
Bram Moolenaar24aa48b2020-07-25 16:33:02 +0200960 CheckDefFailure(['MyDefVarargs("one", "two", 123)'],
Bram Moolenaar77072282020-09-16 17:55:40 +0200961 'E1013: Argument 3: type mismatch, expected string but got number')
Bram Moolenaar24aa48b2020-07-25 16:33:02 +0200962
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +0200963 var lines =<< trim END
Bram Moolenaar24aa48b2020-07-25 16:33:02 +0200964 vim9script
965 def Func(...l: list<string>)
966 echo l
967 enddef
968 Func('a', 'b', 'c')
969 END
970 CheckScriptSuccess(lines)
971
972 lines =<< trim END
973 vim9script
974 def Func(...l: list<string>)
975 echo l
976 enddef
977 Func()
978 END
979 CheckScriptSuccess(lines)
980
981 lines =<< trim END
982 vim9script
Bram Moolenaar2a389082021-04-09 20:24:31 +0200983 def Func(...l: list<any>)
Bram Moolenaar2f8cbc42020-09-16 17:22:59 +0200984 echo l
985 enddef
986 Func(0)
987 END
988 CheckScriptSuccess(lines)
989
990 lines =<< trim END
991 vim9script
Bram Moolenaar2a389082021-04-09 20:24:31 +0200992 def Func(...l: any)
993 echo l
994 enddef
995 Func(0)
996 END
997 CheckScriptFailure(lines, 'E1180:', 2)
998
999 lines =<< trim END
1000 vim9script
Bram Moolenaar28022722020-09-21 22:02:49 +02001001 def Func(..._l: list<string>)
1002 echo _l
1003 enddef
1004 Func('a', 'b', 'c')
1005 END
1006 CheckScriptSuccess(lines)
1007
1008 lines =<< trim END
1009 vim9script
Bram Moolenaar24aa48b2020-07-25 16:33:02 +02001010 def Func(...l: list<string>)
1011 echo l
1012 enddef
1013 Func(1, 2, 3)
1014 END
Bram Moolenaar77072282020-09-16 17:55:40 +02001015 CheckScriptFailure(lines, 'E1013: Argument 1: type mismatch')
Bram Moolenaar24aa48b2020-07-25 16:33:02 +02001016
1017 lines =<< trim END
1018 vim9script
1019 def Func(...l: list<string>)
1020 echo l
1021 enddef
1022 Func('a', 9)
1023 END
Bram Moolenaar77072282020-09-16 17:55:40 +02001024 CheckScriptFailure(lines, 'E1013: Argument 2: type mismatch')
Bram Moolenaar24aa48b2020-07-25 16:33:02 +02001025
1026 lines =<< trim END
1027 vim9script
1028 def Func(...l: list<string>)
1029 echo l
1030 enddef
1031 Func(1, 'a')
1032 END
Bram Moolenaar77072282020-09-16 17:55:40 +02001033 CheckScriptFailure(lines, 'E1013: Argument 1: type mismatch')
Bram Moolenaar4f53b792021-02-07 15:59:49 +01001034
1035 lines =<< trim END
1036 vim9script
1037 def Func( # some comment
1038 ...l = []
1039 )
1040 echo l
1041 enddef
1042 END
1043 CheckScriptFailure(lines, 'E1160:')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001044enddef
1045
Bram Moolenaar1378fbc2020-04-11 20:50:33 +02001046let s:value = ''
1047
1048def FuncOneDefArg(opt = 'text')
1049 s:value = opt
1050enddef
1051
1052def FuncTwoDefArg(nr = 123, opt = 'text'): string
1053 return nr .. opt
1054enddef
1055
1056def FuncVarargs(...arg: list<string>): string
1057 return join(arg, ',')
1058enddef
1059
1060def Test_func_type_varargs()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001061 var RefDefArg: func(?string)
Bram Moolenaar1378fbc2020-04-11 20:50:33 +02001062 RefDefArg = FuncOneDefArg
1063 RefDefArg()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001064 s:value->assert_equal('text')
Bram Moolenaar1378fbc2020-04-11 20:50:33 +02001065 RefDefArg('some')
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001066 s:value->assert_equal('some')
Bram Moolenaar1378fbc2020-04-11 20:50:33 +02001067
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001068 var RefDef2Arg: func(?number, ?string): string
Bram Moolenaar1378fbc2020-04-11 20:50:33 +02001069 RefDef2Arg = FuncTwoDefArg
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001070 RefDef2Arg()->assert_equal('123text')
1071 RefDef2Arg(99)->assert_equal('99text')
1072 RefDef2Arg(77, 'some')->assert_equal('77some')
Bram Moolenaar1378fbc2020-04-11 20:50:33 +02001073
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001074 CheckDefFailure(['var RefWrong: func(string?)'], 'E1010:')
1075 CheckDefFailure(['var RefWrong: func(?string, string)'], 'E1007:')
Bram Moolenaar1378fbc2020-04-11 20:50:33 +02001076
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001077 var RefVarargs: func(...list<string>): string
Bram Moolenaar1378fbc2020-04-11 20:50:33 +02001078 RefVarargs = FuncVarargs
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001079 RefVarargs()->assert_equal('')
1080 RefVarargs('one')->assert_equal('one')
1081 RefVarargs('one', 'two')->assert_equal('one,two')
Bram Moolenaar1378fbc2020-04-11 20:50:33 +02001082
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001083 CheckDefFailure(['var RefWrong: func(...list<string>, string)'], 'E110:')
1084 CheckDefFailure(['var RefWrong: func(...list<string>, ?string)'], 'E110:')
Bram Moolenaar1378fbc2020-04-11 20:50:33 +02001085enddef
1086
Bram Moolenaar0b76b422020-04-07 22:05:08 +02001087" Only varargs
1088def MyVarargsOnly(...args: list<string>): string
1089 return join(args, ',')
1090enddef
1091
1092def Test_call_varargs_only()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001093 MyVarargsOnly()->assert_equal('')
1094 MyVarargsOnly('one')->assert_equal('one')
1095 MyVarargsOnly('one', 'two')->assert_equal('one,two')
Bram Moolenaar77072282020-09-16 17:55:40 +02001096 CheckDefFailure(['MyVarargsOnly(1)'], 'E1013: Argument 1: type mismatch, expected string but got number')
1097 CheckDefFailure(['MyVarargsOnly("one", 2)'], 'E1013: Argument 2: type mismatch, expected string but got number')
Bram Moolenaar0b76b422020-04-07 22:05:08 +02001098enddef
1099
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001100def Test_using_var_as_arg()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001101 writefile(['def Func(x: number)', 'var x = 234', 'enddef', 'defcompile'], 'Xdef')
Bram Moolenaar9bd5d872020-09-06 21:47:48 +02001102 assert_fails('so Xdef', 'E1006:', '', 1, 'Func')
Bram Moolenaard2c61702020-09-06 15:58:36 +02001103 delete('Xdef')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001104enddef
1105
Bram Moolenaarcb2bdb12020-05-10 22:53:56 +02001106def DictArg(arg: dict<string>)
1107 arg['key'] = 'value'
1108enddef
1109
1110def ListArg(arg: list<string>)
1111 arg[0] = 'value'
1112enddef
1113
1114def Test_assign_to_argument()
Bram Moolenaarf5be8cd2020-07-17 20:36:00 +02001115 # works for dict and list
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001116 var d: dict<string> = {}
Bram Moolenaarcb2bdb12020-05-10 22:53:56 +02001117 DictArg(d)
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001118 d['key']->assert_equal('value')
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001119 var l: list<string> = []
Bram Moolenaarcb2bdb12020-05-10 22:53:56 +02001120 ListArg(l)
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001121 l[0]->assert_equal('value')
Bram Moolenaarcb2bdb12020-05-10 22:53:56 +02001122
Bram Moolenaard2c61702020-09-06 15:58:36 +02001123 CheckScriptFailure(['def Func(arg: number)', 'arg = 3', 'enddef', 'defcompile'], 'E1090:')
Bram Moolenaar2d870f82020-12-05 13:41:01 +01001124 delfunc! g:Func
Bram Moolenaarcb2bdb12020-05-10 22:53:56 +02001125enddef
1126
Bram Moolenaarb816dae2020-09-20 22:04:00 +02001127" These argument names are reserved in legacy functions.
1128def WithReservedNames(firstline: string, lastline: string): string
1129 return firstline .. lastline
1130enddef
1131
1132def Test_argument_names()
1133 assert_equal('OK', WithReservedNames('O', 'K'))
1134enddef
1135
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001136def Test_call_func_defined_later()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001137 g:DefinedLater('one')->assert_equal('one')
Bram Moolenaar9bd5d872020-09-06 21:47:48 +02001138 assert_fails('NotDefined("one")', 'E117:', '', 2, 'Test_call_func_defined_later')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001139enddef
1140
Bram Moolenaar1df8b3f2020-04-23 18:13:23 +02001141func DefinedLater(arg)
1142 return a:arg
1143endfunc
1144
1145def Test_call_funcref()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001146 g:SomeFunc('abc')->assert_equal(3)
Bram Moolenaar9bd5d872020-09-06 21:47:48 +02001147 assert_fails('NotAFunc()', 'E117:', '', 2, 'Test_call_funcref') # comment after call
1148 assert_fails('g:NotAFunc()', 'E117:', '', 3, 'Test_call_funcref')
Bram Moolenaar2f1980f2020-07-22 19:30:06 +02001149
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001150 var lines =<< trim END
Bram Moolenaar2f1980f2020-07-22 19:30:06 +02001151 vim9script
1152 def RetNumber(): number
1153 return 123
1154 enddef
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001155 var Funcref: func: number = function('RetNumber')
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001156 Funcref()->assert_equal(123)
Bram Moolenaar2f1980f2020-07-22 19:30:06 +02001157 END
1158 CheckScriptSuccess(lines)
Bram Moolenaar0f60e802020-07-22 20:16:11 +02001159
1160 lines =<< trim END
1161 vim9script
1162 def RetNumber(): number
1163 return 123
1164 enddef
1165 def Bar(F: func: number): number
1166 return F()
1167 enddef
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001168 var Funcref = function('RetNumber')
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001169 Bar(Funcref)->assert_equal(123)
Bram Moolenaar0f60e802020-07-22 20:16:11 +02001170 END
1171 CheckScriptSuccess(lines)
Bram Moolenaarbfba8652020-07-23 20:09:10 +02001172
1173 lines =<< trim END
1174 vim9script
1175 def UseNumber(nr: number)
1176 echo nr
1177 enddef
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001178 var Funcref: func(number) = function('UseNumber')
Bram Moolenaarbfba8652020-07-23 20:09:10 +02001179 Funcref(123)
1180 END
1181 CheckScriptSuccess(lines)
Bram Moolenaarb8070e32020-07-23 20:56:04 +02001182
1183 lines =<< trim END
1184 vim9script
1185 def UseNumber(nr: number)
1186 echo nr
1187 enddef
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001188 var Funcref: func(string) = function('UseNumber')
Bram Moolenaarb8070e32020-07-23 20:56:04 +02001189 END
Bram Moolenaar5e654232020-09-16 15:22:00 +02001190 CheckScriptFailure(lines, 'E1012: Type mismatch; expected func(string) but got func(number)')
Bram Moolenaar4fc224c2020-07-26 17:56:25 +02001191
1192 lines =<< trim END
1193 vim9script
1194 def EchoNr(nr = 34)
1195 g:echo = nr
1196 enddef
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001197 var Funcref: func(?number) = function('EchoNr')
Bram Moolenaar4fc224c2020-07-26 17:56:25 +02001198 Funcref()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001199 g:echo->assert_equal(34)
Bram Moolenaar4fc224c2020-07-26 17:56:25 +02001200 Funcref(123)
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001201 g:echo->assert_equal(123)
Bram Moolenaar4fc224c2020-07-26 17:56:25 +02001202 END
1203 CheckScriptSuccess(lines)
Bram Moolenaarace61322020-07-26 18:16:58 +02001204
1205 lines =<< trim END
1206 vim9script
1207 def EchoList(...l: list<number>)
1208 g:echo = l
1209 enddef
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001210 var Funcref: func(...list<number>) = function('EchoList')
Bram Moolenaarace61322020-07-26 18:16:58 +02001211 Funcref()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001212 g:echo->assert_equal([])
Bram Moolenaarace61322020-07-26 18:16:58 +02001213 Funcref(1, 2, 3)
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001214 g:echo->assert_equal([1, 2, 3])
Bram Moolenaarace61322020-07-26 18:16:58 +02001215 END
1216 CheckScriptSuccess(lines)
Bram Moolenaar01865ad2020-07-26 18:33:09 +02001217
1218 lines =<< trim END
1219 vim9script
1220 def OptAndVar(nr: number, opt = 12, ...l: list<number>): number
1221 g:optarg = opt
1222 g:listarg = l
1223 return nr
1224 enddef
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001225 var Funcref: func(number, ?number, ...list<number>): number = function('OptAndVar')
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001226 Funcref(10)->assert_equal(10)
1227 g:optarg->assert_equal(12)
1228 g:listarg->assert_equal([])
Bram Moolenaar01865ad2020-07-26 18:33:09 +02001229
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001230 Funcref(11, 22)->assert_equal(11)
1231 g:optarg->assert_equal(22)
1232 g:listarg->assert_equal([])
Bram Moolenaar01865ad2020-07-26 18:33:09 +02001233
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001234 Funcref(17, 18, 1, 2, 3)->assert_equal(17)
1235 g:optarg->assert_equal(18)
1236 g:listarg->assert_equal([1, 2, 3])
Bram Moolenaar01865ad2020-07-26 18:33:09 +02001237 END
1238 CheckScriptSuccess(lines)
Bram Moolenaar1df8b3f2020-04-23 18:13:23 +02001239enddef
1240
1241let SomeFunc = function('len')
1242let NotAFunc = 'text'
1243
Bram Moolenaar99aaf0c2020-04-12 14:39:53 +02001244def CombineFuncrefTypes()
Bram Moolenaarf5be8cd2020-07-17 20:36:00 +02001245 # same arguments, different return type
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001246 var Ref1: func(bool): string
1247 var Ref2: func(bool): number
1248 var Ref3: func(bool): any
Bram Moolenaar99aaf0c2020-04-12 14:39:53 +02001249 Ref3 = g:cond ? Ref1 : Ref2
1250
Bram Moolenaarf5be8cd2020-07-17 20:36:00 +02001251 # different number of arguments
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001252 var Refa1: func(bool): number
1253 var Refa2: func(bool, number): number
1254 var Refa3: func: number
Bram Moolenaar99aaf0c2020-04-12 14:39:53 +02001255 Refa3 = g:cond ? Refa1 : Refa2
1256
Bram Moolenaarf5be8cd2020-07-17 20:36:00 +02001257 # different argument types
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001258 var Refb1: func(bool, string): number
1259 var Refb2: func(string, number): number
1260 var Refb3: func(any, any): number
Bram Moolenaar99aaf0c2020-04-12 14:39:53 +02001261 Refb3 = g:cond ? Refb1 : Refb2
1262enddef
1263
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001264def FuncWithForwardCall()
Bram Moolenaar1df8b3f2020-04-23 18:13:23 +02001265 return g:DefinedEvenLater("yes")
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001266enddef
1267
1268def DefinedEvenLater(arg: string): string
1269 return arg
1270enddef
1271
1272def Test_error_in_nested_function()
Bram Moolenaarf5be8cd2020-07-17 20:36:00 +02001273 # Error in called function requires unwinding the call stack.
Bram Moolenaar44d66522020-09-06 22:26:57 +02001274 assert_fails('FuncWithForwardCall()', 'E1096:', '', 1, 'FuncWithForwardCall')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001275enddef
1276
1277def Test_return_type_wrong()
Bram Moolenaar5a849da2020-08-08 16:47:30 +02001278 CheckScriptFailure([
1279 'def Func(): number',
1280 'return "a"',
1281 'enddef',
1282 'defcompile'], 'expected number but got string')
Bram Moolenaar2d870f82020-12-05 13:41:01 +01001283 delfunc! g:Func
Bram Moolenaar5a849da2020-08-08 16:47:30 +02001284 CheckScriptFailure([
1285 'def Func(): string',
1286 'return 1',
1287 'enddef',
1288 'defcompile'], 'expected string but got number')
Bram Moolenaar2d870f82020-12-05 13:41:01 +01001289 delfunc! g:Func
Bram Moolenaar5a849da2020-08-08 16:47:30 +02001290 CheckScriptFailure([
1291 'def Func(): void',
1292 'return "a"',
1293 'enddef',
1294 'defcompile'],
1295 'E1096: Returning a value in a function without a return type')
Bram Moolenaar2d870f82020-12-05 13:41:01 +01001296 delfunc! g:Func
Bram Moolenaar5a849da2020-08-08 16:47:30 +02001297 CheckScriptFailure([
1298 'def Func()',
1299 'return "a"',
1300 'enddef',
1301 'defcompile'],
1302 'E1096: Returning a value in a function without a return type')
Bram Moolenaar2d870f82020-12-05 13:41:01 +01001303 delfunc! g:Func
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001304
Bram Moolenaar5a849da2020-08-08 16:47:30 +02001305 CheckScriptFailure([
1306 'def Func(): number',
1307 'return',
1308 'enddef',
1309 'defcompile'], 'E1003:')
Bram Moolenaar2d870f82020-12-05 13:41:01 +01001310 delfunc! g:Func
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001311
1312 CheckScriptFailure(['def Func(): list', 'return []', 'enddef'], 'E1008:')
Bram Moolenaar2d870f82020-12-05 13:41:01 +01001313 delfunc! g:Func
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001314 CheckScriptFailure(['def Func(): dict', 'return {}', 'enddef'], 'E1008:')
Bram Moolenaar2d870f82020-12-05 13:41:01 +01001315 delfunc! g:Func
Bram Moolenaaree4e0c12020-04-06 21:35:05 +02001316 CheckScriptFailure(['def Func()', 'return 1'], 'E1057:')
Bram Moolenaar2d870f82020-12-05 13:41:01 +01001317 delfunc! g:Func
Bram Moolenaar5a849da2020-08-08 16:47:30 +02001318
1319 CheckScriptFailure([
1320 'vim9script',
1321 'def FuncB()',
1322 ' return 123',
1323 'enddef',
1324 'def FuncA()',
1325 ' FuncB()',
1326 'enddef',
1327 'defcompile'], 'E1096:')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001328enddef
1329
1330def Test_arg_type_wrong()
1331 CheckScriptFailure(['def Func3(items: list)', 'echo "a"', 'enddef'], 'E1008: Missing <type>')
Bram Moolenaaree4e0c12020-04-06 21:35:05 +02001332 CheckScriptFailure(['def Func4(...)', 'echo "a"', 'enddef'], 'E1055: Missing name after ...')
Bram Moolenaarf93c7fe2020-04-23 22:16:53 +02001333 CheckScriptFailure(['def Func5(items:string)', 'echo "a"'], 'E1069:')
Bram Moolenaar6e949782020-04-13 17:21:00 +02001334 CheckScriptFailure(['def Func5(items)', 'echo "a"'], 'E1077:')
Yegappan Lakshmanan34fcb692021-05-25 20:14:00 +02001335 CheckScriptFailure(['def Func6(...x:list<number>)', 'echo "a"', 'enddef'], 'E1069:')
1336 CheckScriptFailure(['def Func7(...x: int)', 'echo "a"', 'enddef'], 'E1010:')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001337enddef
1338
Bram Moolenaar86cdb8a2021-04-06 19:01:03 +02001339def Test_white_space_before_comma()
1340 var lines =<< trim END
1341 vim9script
1342 def Func(a: number , b: number)
1343 enddef
1344 END
1345 CheckScriptFailure(lines, 'E1068:')
Yegappan Lakshmanan611728f2021-05-24 15:15:47 +02001346 call assert_fails('vim9cmd echo stridx("a" .. "b" , "a")', 'E1068:')
Bram Moolenaar86cdb8a2021-04-06 19:01:03 +02001347enddef
1348
Bram Moolenaar608d78f2021-03-06 22:33:12 +01001349def Test_white_space_after_comma()
1350 var lines =<< trim END
1351 vim9script
1352 def Func(a: number,b: number)
1353 enddef
1354 END
1355 CheckScriptFailure(lines, 'E1069:')
1356
1357 # OK in legacy function
1358 lines =<< trim END
1359 vim9script
1360 func Func(a,b)
1361 endfunc
1362 END
1363 CheckScriptSuccess(lines)
1364enddef
1365
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001366def Test_vim9script_call()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001367 var lines =<< trim END
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001368 vim9script
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001369 var name = ''
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001370 def MyFunc(arg: string)
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001371 name = arg
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001372 enddef
1373 MyFunc('foobar')
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001374 name->assert_equal('foobar')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001375
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001376 var str = 'barfoo'
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001377 str->MyFunc()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001378 name->assert_equal('barfoo')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001379
Bram Moolenaar67979662020-06-20 22:50:47 +02001380 g:value = 'value'
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001381 g:value->MyFunc()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001382 name->assert_equal('value')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001383
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001384 var listvar = []
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001385 def ListFunc(arg: list<number>)
1386 listvar = arg
1387 enddef
1388 [1, 2, 3]->ListFunc()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001389 listvar->assert_equal([1, 2, 3])
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001390
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001391 var dictvar = {}
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001392 def DictFunc(arg: dict<number>)
1393 dictvar = arg
1394 enddef
Bram Moolenaare0de1712020-12-02 17:36:54 +01001395 {a: 1, b: 2}->DictFunc()
1396 dictvar->assert_equal({a: 1, b: 2})
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001397 def CompiledDict()
Bram Moolenaare0de1712020-12-02 17:36:54 +01001398 {a: 3, b: 4}->DictFunc()
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001399 enddef
1400 CompiledDict()
Bram Moolenaare0de1712020-12-02 17:36:54 +01001401 dictvar->assert_equal({a: 3, b: 4})
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001402
Bram Moolenaare0de1712020-12-02 17:36:54 +01001403 {a: 3, b: 4}->DictFunc()
1404 dictvar->assert_equal({a: 3, b: 4})
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001405
1406 ('text')->MyFunc()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001407 name->assert_equal('text')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001408 ("some")->MyFunc()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001409 name->assert_equal('some')
Bram Moolenaare6b53242020-07-01 17:28:33 +02001410
Bram Moolenaar13e12b82020-07-24 18:47:22 +02001411 # line starting with single quote is not a mark
Bram Moolenaar10409562020-07-29 20:00:38 +02001412 # line starting with double quote can be a method call
Bram Moolenaar3d48e252020-07-15 14:15:52 +02001413 'asdfasdf'->MyFunc()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001414 name->assert_equal('asdfasdf')
Bram Moolenaar10409562020-07-29 20:00:38 +02001415 "xyz"->MyFunc()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001416 name->assert_equal('xyz')
Bram Moolenaar3d48e252020-07-15 14:15:52 +02001417
1418 def UseString()
1419 'xyork'->MyFunc()
1420 enddef
1421 UseString()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001422 name->assert_equal('xyork')
Bram Moolenaar3d48e252020-07-15 14:15:52 +02001423
Bram Moolenaar10409562020-07-29 20:00:38 +02001424 def UseString2()
1425 "knife"->MyFunc()
1426 enddef
1427 UseString2()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001428 name->assert_equal('knife')
Bram Moolenaar10409562020-07-29 20:00:38 +02001429
Bram Moolenaar13e12b82020-07-24 18:47:22 +02001430 # prepending a colon makes it a mark
1431 new
1432 setline(1, ['aaa', 'bbb', 'ccc'])
1433 normal! 3Gmt1G
1434 :'t
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001435 getcurpos()[1]->assert_equal(3)
Bram Moolenaar13e12b82020-07-24 18:47:22 +02001436 bwipe!
1437
Bram Moolenaare6b53242020-07-01 17:28:33 +02001438 MyFunc(
1439 'continued'
1440 )
1441 assert_equal('continued',
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001442 name
Bram Moolenaare6b53242020-07-01 17:28:33 +02001443 )
1444
1445 call MyFunc(
1446 'more'
1447 ..
1448 'lines'
1449 )
1450 assert_equal(
1451 'morelines',
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001452 name)
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001453 END
1454 writefile(lines, 'Xcall.vim')
1455 source Xcall.vim
1456 delete('Xcall.vim')
1457enddef
1458
1459def Test_vim9script_call_fail_decl()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001460 var lines =<< trim END
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001461 vim9script
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001462 var name = ''
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001463 def MyFunc(arg: string)
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001464 var name = 123
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001465 enddef
Bram Moolenaar822ba242020-05-24 23:00:18 +02001466 defcompile
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001467 END
Bram Moolenaar6c4bfe42020-07-23 18:26:30 +02001468 CheckScriptFailure(lines, 'E1054:')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001469enddef
1470
Bram Moolenaar65b95452020-07-19 14:03:09 +02001471def Test_vim9script_call_fail_type()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001472 var lines =<< trim END
Bram Moolenaar65b95452020-07-19 14:03:09 +02001473 vim9script
1474 def MyFunc(arg: string)
1475 echo arg
1476 enddef
1477 MyFunc(1234)
1478 END
Bram Moolenaar77072282020-09-16 17:55:40 +02001479 CheckScriptFailure(lines, 'E1013: Argument 1: type mismatch, expected string but got number')
Bram Moolenaar65b95452020-07-19 14:03:09 +02001480enddef
1481
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001482def Test_vim9script_call_fail_const()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001483 var lines =<< trim END
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001484 vim9script
1485 const var = ''
1486 def MyFunc(arg: string)
1487 var = 'asdf'
1488 enddef
Bram Moolenaar822ba242020-05-24 23:00:18 +02001489 defcompile
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001490 END
1491 writefile(lines, 'Xcall_const.vim')
Bram Moolenaar9bd5d872020-09-06 21:47:48 +02001492 assert_fails('source Xcall_const.vim', 'E46:', '', 1, 'MyFunc')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001493 delete('Xcall_const.vim')
Bram Moolenaar3bdc90b2020-12-22 20:35:40 +01001494
1495 lines =<< trim END
1496 const g:Aconst = 77
1497 def Change()
1498 # comment
1499 g:Aconst = 99
1500 enddef
1501 call Change()
1502 unlet g:Aconst
1503 END
Bram Moolenaar1dcf55d2020-12-22 22:07:30 +01001504 CheckScriptFailure(lines, 'E741: Value is locked: Aconst', 2)
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001505enddef
1506
1507" Test that inside :function a Python function can be defined, :def is not
1508" recognized.
1509func Test_function_python()
1510 CheckFeature python3
Bram Moolenaar727345e2020-09-27 23:33:59 +02001511 let py = 'python3'
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001512 execute py "<< EOF"
1513def do_something():
1514 return 1
1515EOF
1516endfunc
1517
1518def Test_delfunc()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001519 var lines =<< trim END
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001520 vim9script
Bram Moolenaar4c17ad92020-04-27 22:47:51 +02001521 def g:GoneSoon()
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001522 echo 'hello'
1523 enddef
1524
1525 def CallGoneSoon()
1526 GoneSoon()
1527 enddef
Bram Moolenaar822ba242020-05-24 23:00:18 +02001528 defcompile
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001529
Bram Moolenaar4c17ad92020-04-27 22:47:51 +02001530 delfunc g:GoneSoon
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001531 CallGoneSoon()
1532 END
1533 writefile(lines, 'XToDelFunc')
Bram Moolenaar9bd5d872020-09-06 21:47:48 +02001534 assert_fails('so XToDelFunc', 'E933:', '', 1, 'CallGoneSoon')
1535 assert_fails('so XToDelFunc', 'E933:', '', 1, 'CallGoneSoon')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001536
1537 delete('XToDelFunc')
1538enddef
1539
1540def Test_redef_failure()
Bram Moolenaard2c61702020-09-06 15:58:36 +02001541 writefile(['def Func0(): string', 'return "Func0"', 'enddef'], 'Xdef')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001542 so Xdef
Bram Moolenaard2c61702020-09-06 15:58:36 +02001543 writefile(['def Func1(): string', 'return "Func1"', 'enddef'], 'Xdef')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001544 so Xdef
Bram Moolenaard2c61702020-09-06 15:58:36 +02001545 writefile(['def! Func0(): string', 'enddef', 'defcompile'], 'Xdef')
Bram Moolenaar9bd5d872020-09-06 21:47:48 +02001546 assert_fails('so Xdef', 'E1027:', '', 1, 'Func0')
Bram Moolenaard2c61702020-09-06 15:58:36 +02001547 writefile(['def Func2(): string', 'return "Func2"', 'enddef'], 'Xdef')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001548 so Xdef
Bram Moolenaard2c61702020-09-06 15:58:36 +02001549 delete('Xdef')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001550
Bram Moolenaar701cc6c2021-04-10 13:33:48 +02001551 assert_fails('g:Func0()', 'E1091:')
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001552 g:Func1()->assert_equal('Func1')
1553 g:Func2()->assert_equal('Func2')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001554
1555 delfunc! Func0
1556 delfunc! Func1
1557 delfunc! Func2
1558enddef
1559
Bram Moolenaarf93c7fe2020-04-23 22:16:53 +02001560def Test_vim9script_func()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001561 var lines =<< trim END
Bram Moolenaarf93c7fe2020-04-23 22:16:53 +02001562 vim9script
1563 func Func(arg)
1564 echo a:arg
1565 endfunc
1566 Func('text')
1567 END
1568 writefile(lines, 'XVim9Func')
1569 so XVim9Func
1570
1571 delete('XVim9Func')
1572enddef
1573
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001574let s:funcResult = 0
1575
1576def FuncNoArgNoRet()
Bram Moolenaar53900992020-08-22 19:02:02 +02001577 s:funcResult = 11
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001578enddef
1579
1580def FuncNoArgRetNumber(): number
Bram Moolenaar53900992020-08-22 19:02:02 +02001581 s:funcResult = 22
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001582 return 1234
1583enddef
1584
Bram Moolenaarec5929d2020-04-07 20:53:39 +02001585def FuncNoArgRetString(): string
Bram Moolenaar53900992020-08-22 19:02:02 +02001586 s:funcResult = 45
Bram Moolenaarec5929d2020-04-07 20:53:39 +02001587 return 'text'
1588enddef
1589
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001590def FuncOneArgNoRet(arg: number)
Bram Moolenaar53900992020-08-22 19:02:02 +02001591 s:funcResult = arg
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001592enddef
1593
1594def FuncOneArgRetNumber(arg: number): number
Bram Moolenaar53900992020-08-22 19:02:02 +02001595 s:funcResult = arg
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001596 return arg
1597enddef
1598
Bram Moolenaar08938ee2020-04-11 23:17:17 +02001599def FuncTwoArgNoRet(one: bool, two: number)
Bram Moolenaar53900992020-08-22 19:02:02 +02001600 s:funcResult = two
Bram Moolenaar08938ee2020-04-11 23:17:17 +02001601enddef
1602
Bram Moolenaarec5929d2020-04-07 20:53:39 +02001603def FuncOneArgRetString(arg: string): string
1604 return arg
1605enddef
1606
Bram Moolenaar89228602020-04-05 22:14:54 +02001607def FuncOneArgRetAny(arg: any): any
1608 return arg
1609enddef
1610
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001611def Test_func_type()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001612 var Ref1: func()
Bram Moolenaar53900992020-08-22 19:02:02 +02001613 s:funcResult = 0
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001614 Ref1 = FuncNoArgNoRet
1615 Ref1()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001616 s:funcResult->assert_equal(11)
Bram Moolenaar4c683752020-04-05 21:38:23 +02001617
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001618 var Ref2: func
Bram Moolenaar53900992020-08-22 19:02:02 +02001619 s:funcResult = 0
Bram Moolenaar4c683752020-04-05 21:38:23 +02001620 Ref2 = FuncNoArgNoRet
1621 Ref2()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001622 s:funcResult->assert_equal(11)
Bram Moolenaar4c683752020-04-05 21:38:23 +02001623
Bram Moolenaar53900992020-08-22 19:02:02 +02001624 s:funcResult = 0
Bram Moolenaar4c683752020-04-05 21:38:23 +02001625 Ref2 = FuncOneArgNoRet
1626 Ref2(12)
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001627 s:funcResult->assert_equal(12)
Bram Moolenaar4c683752020-04-05 21:38:23 +02001628
Bram Moolenaar53900992020-08-22 19:02:02 +02001629 s:funcResult = 0
Bram Moolenaar4c683752020-04-05 21:38:23 +02001630 Ref2 = FuncNoArgRetNumber
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001631 Ref2()->assert_equal(1234)
1632 s:funcResult->assert_equal(22)
Bram Moolenaar4c683752020-04-05 21:38:23 +02001633
Bram Moolenaar53900992020-08-22 19:02:02 +02001634 s:funcResult = 0
Bram Moolenaar4c683752020-04-05 21:38:23 +02001635 Ref2 = FuncOneArgRetNumber
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001636 Ref2(13)->assert_equal(13)
1637 s:funcResult->assert_equal(13)
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001638enddef
1639
Bram Moolenaar9978d472020-07-05 16:01:56 +02001640def Test_repeat_return_type()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001641 var res = 0
Bram Moolenaar9978d472020-07-05 16:01:56 +02001642 for n in repeat([1], 3)
1643 res += n
1644 endfor
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001645 res->assert_equal(3)
Bram Moolenaarfce82b32020-07-05 16:07:21 +02001646
1647 res = 0
1648 for n in add([1, 2], 3)
1649 res += n
1650 endfor
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001651 res->assert_equal(6)
Bram Moolenaar9978d472020-07-05 16:01:56 +02001652enddef
1653
Bram Moolenaar846178a2020-07-05 17:04:13 +02001654def Test_argv_return_type()
1655 next fileone filetwo
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001656 var res = ''
Bram Moolenaar846178a2020-07-05 17:04:13 +02001657 for name in argv()
1658 res ..= name
1659 endfor
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001660 res->assert_equal('fileonefiletwo')
Bram Moolenaar846178a2020-07-05 17:04:13 +02001661enddef
1662
Bram Moolenaarec5929d2020-04-07 20:53:39 +02001663def Test_func_type_part()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001664 var RefVoid: func: void
Bram Moolenaarec5929d2020-04-07 20:53:39 +02001665 RefVoid = FuncNoArgNoRet
1666 RefVoid = FuncOneArgNoRet
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001667 CheckDefFailure(['var RefVoid: func: void', 'RefVoid = FuncNoArgRetNumber'], 'E1012: Type mismatch; expected func(...) but got func(): number')
1668 CheckDefFailure(['var RefVoid: func: void', 'RefVoid = FuncNoArgRetString'], 'E1012: Type mismatch; expected func(...) but got func(): string')
Bram Moolenaarec5929d2020-04-07 20:53:39 +02001669
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001670 var RefAny: func(): any
Bram Moolenaarec5929d2020-04-07 20:53:39 +02001671 RefAny = FuncNoArgRetNumber
1672 RefAny = FuncNoArgRetString
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001673 CheckDefFailure(['var RefAny: func(): any', 'RefAny = FuncNoArgNoRet'], 'E1012: Type mismatch; expected func(): any but got func()')
1674 CheckDefFailure(['var RefAny: func(): any', 'RefAny = FuncOneArgNoRet'], 'E1012: Type mismatch; expected func(): any but got func(number)')
Bram Moolenaarec5929d2020-04-07 20:53:39 +02001675
Bram Moolenaar6abd3dc2020-10-04 14:17:32 +02001676 var RefAnyNoArgs: func: any = RefAny
1677
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001678 var RefNr: func: number
Bram Moolenaarec5929d2020-04-07 20:53:39 +02001679 RefNr = FuncNoArgRetNumber
1680 RefNr = FuncOneArgRetNumber
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001681 CheckDefFailure(['var RefNr: func: number', 'RefNr = FuncNoArgNoRet'], 'E1012: Type mismatch; expected func(...): number but got func()')
1682 CheckDefFailure(['var RefNr: func: number', 'RefNr = FuncNoArgRetString'], 'E1012: Type mismatch; expected func(...): number but got func(): string')
Bram Moolenaarec5929d2020-04-07 20:53:39 +02001683
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001684 var RefStr: func: string
Bram Moolenaarec5929d2020-04-07 20:53:39 +02001685 RefStr = FuncNoArgRetString
1686 RefStr = FuncOneArgRetString
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001687 CheckDefFailure(['var RefStr: func: string', 'RefStr = FuncNoArgNoRet'], 'E1012: Type mismatch; expected func(...): string but got func()')
1688 CheckDefFailure(['var RefStr: func: string', 'RefStr = FuncNoArgRetNumber'], 'E1012: Type mismatch; expected func(...): string but got func(): number')
Bram Moolenaarec5929d2020-04-07 20:53:39 +02001689enddef
1690
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001691def Test_func_type_fails()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001692 CheckDefFailure(['var ref1: func()'], 'E704:')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001693
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001694 CheckDefFailure(['var Ref1: func()', 'Ref1 = FuncNoArgRetNumber'], 'E1012: Type mismatch; expected func() but got func(): number')
1695 CheckDefFailure(['var Ref1: func()', 'Ref1 = FuncOneArgNoRet'], 'E1012: Type mismatch; expected func() but got func(number)')
1696 CheckDefFailure(['var Ref1: func()', 'Ref1 = FuncOneArgRetNumber'], 'E1012: Type mismatch; expected func() but got func(number): number')
1697 CheckDefFailure(['var Ref1: func(bool)', 'Ref1 = FuncTwoArgNoRet'], 'E1012: Type mismatch; expected func(bool) but got func(bool, number)')
1698 CheckDefFailure(['var Ref1: func(?bool)', 'Ref1 = FuncTwoArgNoRet'], 'E1012: Type mismatch; expected func(?bool) but got func(bool, number)')
1699 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 +02001700
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001701 CheckDefFailure(['var RefWrong: func(string ,number)'], 'E1068:')
1702 CheckDefFailure(['var RefWrong: func(string,number)'], 'E1069:')
1703 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:')
1704 CheckDefFailure(['var RefWrong: func(bool):string'], 'E1069:')
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001705enddef
1706
Bram Moolenaar89228602020-04-05 22:14:54 +02001707def Test_func_return_type()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001708 var nr: number
Bram Moolenaar89228602020-04-05 22:14:54 +02001709 nr = FuncNoArgRetNumber()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001710 nr->assert_equal(1234)
Bram Moolenaar89228602020-04-05 22:14:54 +02001711
1712 nr = FuncOneArgRetAny(122)
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001713 nr->assert_equal(122)
Bram Moolenaar89228602020-04-05 22:14:54 +02001714
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001715 var str: string
Bram Moolenaar89228602020-04-05 22:14:54 +02001716 str = FuncOneArgRetAny('yes')
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001717 str->assert_equal('yes')
Bram Moolenaar89228602020-04-05 22:14:54 +02001718
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001719 CheckDefFailure(['var str: string', 'str = FuncNoArgRetNumber()'], 'E1012: Type mismatch; expected string but got number')
Bram Moolenaar89228602020-04-05 22:14:54 +02001720enddef
1721
Bram Moolenaar6abd3dc2020-10-04 14:17:32 +02001722def Test_func_common_type()
1723 def FuncOne(n: number): number
1724 return n
1725 enddef
1726 def FuncTwo(s: string): number
1727 return len(s)
1728 enddef
1729 def FuncThree(n: number, s: string): number
1730 return n + len(s)
1731 enddef
1732 var list = [FuncOne, FuncTwo, FuncThree]
1733 assert_equal(8, list[0](8))
1734 assert_equal(4, list[1]('word'))
1735 assert_equal(7, list[2](3, 'word'))
1736enddef
1737
Bram Moolenaar5e774c72020-04-12 21:53:00 +02001738def MultiLine(
1739 arg1: string,
1740 arg2 = 1234,
1741 ...rest: list<string>
1742 ): string
1743 return arg1 .. arg2 .. join(rest, '-')
1744enddef
1745
Bram Moolenaar2c330432020-04-13 14:41:35 +02001746def MultiLineComment(
1747 arg1: string, # comment
1748 arg2 = 1234, # comment
1749 ...rest: list<string> # comment
1750 ): string # comment
1751 return arg1 .. arg2 .. join(rest, '-')
1752enddef
1753
Bram Moolenaar5e774c72020-04-12 21:53:00 +02001754def Test_multiline()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001755 MultiLine('text')->assert_equal('text1234')
1756 MultiLine('text', 777)->assert_equal('text777')
1757 MultiLine('text', 777, 'one')->assert_equal('text777one')
1758 MultiLine('text', 777, 'one', 'two')->assert_equal('text777one-two')
Bram Moolenaar5e774c72020-04-12 21:53:00 +02001759enddef
1760
Bram Moolenaar23e03252020-04-12 22:22:31 +02001761func Test_multiline_not_vim9()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001762 call MultiLine('text')->assert_equal('text1234')
1763 call MultiLine('text', 777)->assert_equal('text777')
1764 call MultiLine('text', 777, 'one')->assert_equal('text777one')
1765 call MultiLine('text', 777, 'one', 'two')->assert_equal('text777one-two')
Bram Moolenaar23e03252020-04-12 22:22:31 +02001766endfunc
1767
Bram Moolenaar5e774c72020-04-12 21:53:00 +02001768
Bram Moolenaaree4e0c12020-04-06 21:35:05 +02001769" When using CheckScriptFailure() for the below test, E1010 is generated instead
1770" of E1056.
1771func Test_E1056_1059()
1772 let caught_1056 = 0
1773 try
1774 def F():
1775 return 1
1776 enddef
1777 catch /E1056:/
1778 let caught_1056 = 1
1779 endtry
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001780 eval caught_1056->assert_equal(1)
Bram Moolenaaree4e0c12020-04-06 21:35:05 +02001781
1782 let caught_1059 = 0
1783 try
1784 def F5(items : list)
1785 echo 'a'
1786 enddef
1787 catch /E1059:/
1788 let caught_1059 = 1
1789 endtry
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001790 eval caught_1059->assert_equal(1)
Bram Moolenaaree4e0c12020-04-06 21:35:05 +02001791endfunc
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02001792
Bram Moolenaar015f4262020-05-05 21:25:22 +02001793func DelMe()
1794 echo 'DelMe'
1795endfunc
1796
Bram Moolenaarbf8feb52020-08-08 14:26:31 +02001797def Test_error_reporting()
1798 # comment lines at the start of the function
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001799 var lines =<< trim END
Bram Moolenaarbf8feb52020-08-08 14:26:31 +02001800 " comment
1801 def Func()
1802 # comment
1803 # comment
1804 invalid
1805 enddef
1806 defcompile
1807 END
Bram Moolenaar08052222020-09-14 17:04:31 +02001808 writefile(lines, 'Xdef')
Bram Moolenaarbf8feb52020-08-08 14:26:31 +02001809 try
1810 source Xdef
Bram Moolenaar7517ffd2020-08-14 18:35:07 +02001811 assert_report('should have failed')
Bram Moolenaarbf8feb52020-08-08 14:26:31 +02001812 catch /E476:/
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001813 v:exception->assert_match('Invalid command: invalid')
1814 v:throwpoint->assert_match(', line 3$')
Bram Moolenaarbf8feb52020-08-08 14:26:31 +02001815 endtry
Bram Moolenaar2d870f82020-12-05 13:41:01 +01001816 delfunc! g:Func
Bram Moolenaarbf8feb52020-08-08 14:26:31 +02001817
1818 # comment lines after the start of the function
1819 lines =<< trim END
1820 " comment
1821 def Func()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001822 var x = 1234
Bram Moolenaarbf8feb52020-08-08 14:26:31 +02001823 # comment
1824 # comment
1825 invalid
1826 enddef
1827 defcompile
1828 END
Bram Moolenaar08052222020-09-14 17:04:31 +02001829 writefile(lines, 'Xdef')
Bram Moolenaarbf8feb52020-08-08 14:26:31 +02001830 try
1831 source Xdef
Bram Moolenaar7517ffd2020-08-14 18:35:07 +02001832 assert_report('should have failed')
Bram Moolenaarbf8feb52020-08-08 14:26:31 +02001833 catch /E476:/
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001834 v:exception->assert_match('Invalid command: invalid')
1835 v:throwpoint->assert_match(', line 4$')
Bram Moolenaarbf8feb52020-08-08 14:26:31 +02001836 endtry
Bram Moolenaar2d870f82020-12-05 13:41:01 +01001837 delfunc! g:Func
Bram Moolenaarbf8feb52020-08-08 14:26:31 +02001838
Bram Moolenaar7517ffd2020-08-14 18:35:07 +02001839 lines =<< trim END
1840 vim9script
1841 def Func()
Bram Moolenaare0de1712020-12-02 17:36:54 +01001842 var db = {foo: 1, bar: 2}
Bram Moolenaar7517ffd2020-08-14 18:35:07 +02001843 # comment
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001844 var x = db.asdf
Bram Moolenaar7517ffd2020-08-14 18:35:07 +02001845 enddef
1846 defcompile
1847 Func()
1848 END
Bram Moolenaar08052222020-09-14 17:04:31 +02001849 writefile(lines, 'Xdef')
Bram Moolenaar7517ffd2020-08-14 18:35:07 +02001850 try
1851 source Xdef
1852 assert_report('should have failed')
1853 catch /E716:/
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001854 v:throwpoint->assert_match('_Func, line 3$')
Bram Moolenaar7517ffd2020-08-14 18:35:07 +02001855 endtry
Bram Moolenaar2d870f82020-12-05 13:41:01 +01001856 delfunc! g:Func
Bram Moolenaar7517ffd2020-08-14 18:35:07 +02001857
Bram Moolenaar08052222020-09-14 17:04:31 +02001858 delete('Xdef')
Bram Moolenaarbf8feb52020-08-08 14:26:31 +02001859enddef
1860
Bram Moolenaar015f4262020-05-05 21:25:22 +02001861def Test_deleted_function()
1862 CheckDefExecFailure([
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001863 'var RefMe: func = function("g:DelMe")',
Bram Moolenaar015f4262020-05-05 21:25:22 +02001864 'delfunc g:DelMe',
1865 'echo RefMe()'], 'E117:')
1866enddef
1867
1868def Test_unknown_function()
1869 CheckDefExecFailure([
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001870 'var Ref: func = function("NotExist")',
Bram Moolenaar9b7bf9e2020-07-11 22:14:59 +02001871 'delfunc g:NotExist'], 'E700:')
Bram Moolenaar015f4262020-05-05 21:25:22 +02001872enddef
1873
Bram Moolenaar328eac22021-01-07 19:23:08 +01001874def RefFunc(Ref: func(any): any): string
Bram Moolenaarc8cd2b32020-05-01 19:29:08 +02001875 return Ref('more')
1876enddef
1877
1878def Test_closure_simple()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001879 var local = 'some '
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01001880 RefFunc((s) => local .. s)->assert_equal('some more')
Bram Moolenaarc8cd2b32020-05-01 19:29:08 +02001881enddef
1882
Bram Moolenaarbf67ea12020-05-02 17:52:42 +02001883def MakeRef()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001884 var local = 'some '
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01001885 g:Ref = (s) => local .. s
Bram Moolenaarbf67ea12020-05-02 17:52:42 +02001886enddef
1887
1888def Test_closure_ref_after_return()
1889 MakeRef()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001890 g:Ref('thing')->assert_equal('some thing')
Bram Moolenaarbf67ea12020-05-02 17:52:42 +02001891 unlet g:Ref
1892enddef
1893
Bram Moolenaar5adc55c2020-05-02 23:12:58 +02001894def MakeTwoRefs()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001895 var local = ['some']
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01001896 g:Extend = (s) => local->add(s)
1897 g:Read = () => local
Bram Moolenaar5adc55c2020-05-02 23:12:58 +02001898enddef
1899
1900def Test_closure_two_refs()
1901 MakeTwoRefs()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001902 join(g:Read(), ' ')->assert_equal('some')
Bram Moolenaar5adc55c2020-05-02 23:12:58 +02001903 g:Extend('more')
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001904 join(g:Read(), ' ')->assert_equal('some more')
Bram Moolenaar5adc55c2020-05-02 23:12:58 +02001905 g:Extend('even')
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001906 join(g:Read(), ' ')->assert_equal('some more even')
Bram Moolenaar5adc55c2020-05-02 23:12:58 +02001907
1908 unlet g:Extend
1909 unlet g:Read
1910enddef
1911
Bram Moolenaar5adc55c2020-05-02 23:12:58 +02001912def ReadRef(Ref: func(): list<string>): string
1913 return join(Ref(), ' ')
1914enddef
1915
Bram Moolenaar5e654232020-09-16 15:22:00 +02001916def ExtendRef(Ref: func(string): list<string>, add: string)
Bram Moolenaar5adc55c2020-05-02 23:12:58 +02001917 Ref(add)
1918enddef
1919
1920def Test_closure_two_indirect_refs()
Bram Moolenaarf7779c62020-05-03 15:38:16 +02001921 MakeTwoRefs()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001922 ReadRef(g:Read)->assert_equal('some')
Bram Moolenaar5adc55c2020-05-02 23:12:58 +02001923 ExtendRef(g:Extend, 'more')
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001924 ReadRef(g:Read)->assert_equal('some more')
Bram Moolenaar5adc55c2020-05-02 23:12:58 +02001925 ExtendRef(g:Extend, 'even')
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001926 ReadRef(g:Read)->assert_equal('some more even')
Bram Moolenaar5adc55c2020-05-02 23:12:58 +02001927
1928 unlet g:Extend
1929 unlet g:Read
1930enddef
Bram Moolenaarbf67ea12020-05-02 17:52:42 +02001931
Bram Moolenaar2fd4cd72020-05-03 22:30:49 +02001932def MakeArgRefs(theArg: string)
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001933 var local = 'loc_val'
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01001934 g:UseArg = (s) => theArg .. '/' .. local .. '/' .. s
Bram Moolenaar2fd4cd72020-05-03 22:30:49 +02001935enddef
1936
1937def MakeArgRefsVarargs(theArg: string, ...rest: list<string>)
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001938 var local = 'the_loc'
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01001939 g:UseVararg = (s) => theArg .. '/' .. local .. '/' .. s .. '/' .. join(rest)
Bram Moolenaar2fd4cd72020-05-03 22:30:49 +02001940enddef
1941
1942def Test_closure_using_argument()
1943 MakeArgRefs('arg_val')
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001944 g:UseArg('call_val')->assert_equal('arg_val/loc_val/call_val')
Bram Moolenaar2fd4cd72020-05-03 22:30:49 +02001945
1946 MakeArgRefsVarargs('arg_val', 'one', 'two')
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001947 g:UseVararg('call_val')->assert_equal('arg_val/the_loc/call_val/one two')
Bram Moolenaar2fd4cd72020-05-03 22:30:49 +02001948
1949 unlet g:UseArg
1950 unlet g:UseVararg
Bram Moolenaar44ec21c2021-02-12 21:50:57 +01001951
1952 var lines =<< trim END
1953 vim9script
1954 def Test(Fun: func(number): number): list<number>
1955 return map([1, 2, 3], (_, i) => Fun(i))
1956 enddef
1957 def Inc(nr: number): number
1958 return nr + 2
1959 enddef
1960 assert_equal([3, 4, 5], Test(Inc))
1961 END
1962 CheckScriptSuccess(lines)
Bram Moolenaar2fd4cd72020-05-03 22:30:49 +02001963enddef
1964
Bram Moolenaar85d5e2b2020-10-10 14:13:01 +02001965def MakeGetAndAppendRefs()
1966 var local = 'a'
1967
1968 def Append(arg: string)
1969 local ..= arg
1970 enddef
1971 g:Append = Append
1972
1973 def Get(): string
1974 return local
1975 enddef
1976 g:Get = Get
1977enddef
1978
1979def Test_closure_append_get()
1980 MakeGetAndAppendRefs()
1981 g:Get()->assert_equal('a')
1982 g:Append('-b')
1983 g:Get()->assert_equal('a-b')
1984 g:Append('-c')
1985 g:Get()->assert_equal('a-b-c')
1986
1987 unlet g:Append
1988 unlet g:Get
1989enddef
Bram Moolenaarb68b3462020-05-06 21:06:30 +02001990
Bram Moolenaar04b12692020-05-04 23:24:44 +02001991def Test_nested_closure()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02001992 var local = 'text'
Bram Moolenaar04b12692020-05-04 23:24:44 +02001993 def Closure(arg: string): string
1994 return local .. arg
1995 enddef
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02001996 Closure('!!!')->assert_equal('text!!!')
Bram Moolenaar04b12692020-05-04 23:24:44 +02001997enddef
1998
Bram Moolenaar6f5b6df2020-05-16 21:20:12 +02001999func GetResult(Ref)
2000 return a:Ref('some')
2001endfunc
2002
2003def Test_call_closure_not_compiled()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002004 var text = 'text'
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01002005 g:Ref = (s) => s .. text
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002006 GetResult(g:Ref)->assert_equal('sometext')
Bram Moolenaar6f5b6df2020-05-16 21:20:12 +02002007enddef
2008
Bram Moolenaar7cbfaa52020-09-18 21:25:32 +02002009def Test_double_closure_fails()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002010 var lines =<< trim END
Bram Moolenaar7cbfaa52020-09-18 21:25:32 +02002011 vim9script
2012 def Func()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002013 var name = 0
2014 for i in range(2)
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01002015 timer_start(0, () => name)
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002016 endfor
Bram Moolenaar7cbfaa52020-09-18 21:25:32 +02002017 enddef
2018 Func()
2019 END
Bram Moolenaar148ce7a2020-09-23 21:57:23 +02002020 CheckScriptSuccess(lines)
Bram Moolenaar7cbfaa52020-09-18 21:25:32 +02002021enddef
2022
Bram Moolenaar85d5e2b2020-10-10 14:13:01 +02002023def Test_nested_closure_used()
2024 var lines =<< trim END
2025 vim9script
2026 def Func()
2027 var x = 'hello'
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01002028 var Closure = () => x
2029 g:Myclosure = () => Closure()
Bram Moolenaar85d5e2b2020-10-10 14:13:01 +02002030 enddef
2031 Func()
2032 assert_equal('hello', g:Myclosure())
2033 END
2034 CheckScriptSuccess(lines)
2035enddef
Bram Moolenaar0876c782020-10-07 19:08:04 +02002036
Bram Moolenaarc70bdab2020-09-26 19:59:38 +02002037def Test_nested_closure_fails()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002038 var lines =<< trim END
Bram Moolenaarc70bdab2020-09-26 19:59:38 +02002039 vim9script
2040 def FuncA()
2041 FuncB(0)
2042 enddef
2043 def FuncB(n: number): list<string>
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01002044 return map([0], (_, v) => n)
Bram Moolenaarc70bdab2020-09-26 19:59:38 +02002045 enddef
2046 FuncA()
2047 END
2048 CheckScriptFailure(lines, 'E1012:')
2049enddef
2050
Bram Moolenaarf112f302020-12-20 17:47:52 +01002051def Test_global_closure()
2052 var lines =<< trim END
2053 vim9script
2054 def ReverseEveryNLines(n: number, line1: number, line2: number)
2055 var mods = 'sil keepj keepp lockm '
2056 var range = ':' .. line1 .. ',' .. line2
2057 def g:Offset(): number
2058 var offset = (line('.') - line1 + 1) % n
2059 return offset != 0 ? offset : n
2060 enddef
2061 exe mods .. range .. 'g/^/exe "m .-" .. g:Offset()'
2062 enddef
2063
2064 new
2065 repeat(['aaa', 'bbb', 'ccc'], 3)->setline(1)
2066 ReverseEveryNLines(3, 1, 9)
2067 END
2068 CheckScriptSuccess(lines)
2069 var expected = repeat(['ccc', 'bbb', 'aaa'], 3)
2070 assert_equal(expected, getline(1, 9))
2071 bwipe!
2072enddef
2073
Bram Moolenaarcd45ed02020-12-22 17:35:54 +01002074def Test_global_closure_called_directly()
2075 var lines =<< trim END
2076 vim9script
2077 def Outer()
2078 var x = 1
2079 def g:Inner()
2080 var y = x
2081 x += 1
2082 assert_equal(1, y)
2083 enddef
2084 g:Inner()
2085 assert_equal(2, x)
2086 enddef
2087 Outer()
2088 END
2089 CheckScriptSuccess(lines)
2090 delfunc g:Inner
2091enddef
2092
Bram Moolenaar34c54eb2020-11-25 19:15:19 +01002093def Test_failure_in_called_function()
2094 # this was using the frame index as the return value
2095 var lines =<< trim END
2096 vim9script
2097 au TerminalWinOpen * eval [][0]
2098 def PopupTerm(a: any)
2099 # make sure typvals on stack are string
2100 ['a', 'b', 'c', 'd', 'e', 'f', 'g']->join()
2101 FireEvent()
2102 enddef
2103 def FireEvent()
2104 do TerminalWinOpen
2105 enddef
2106 # use try/catch to make eval fail
2107 try
2108 call PopupTerm(0)
2109 catch
2110 endtry
2111 au! TerminalWinOpen
2112 END
2113 CheckScriptSuccess(lines)
2114enddef
2115
Bram Moolenaar5366e1a2020-10-01 13:01:34 +02002116def Test_nested_lambda()
2117 var lines =<< trim END
2118 vim9script
2119 def Func()
2120 var x = 4
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01002121 var Lambda1 = () => 7
2122 var Lambda2 = () => [Lambda1(), x]
Bram Moolenaar5366e1a2020-10-01 13:01:34 +02002123 var res = Lambda2()
2124 assert_equal([7, 4], res)
2125 enddef
2126 Func()
2127 END
2128 CheckScriptSuccess(lines)
2129enddef
2130
Bram Moolenaar074f84c2021-05-18 11:47:44 +02002131def Test_nested_inline_lambda()
2132 # TODO: use the "text" argument
2133 var lines =<< trim END
2134 vim9script
2135 def F(text: string): func(string): func(string): string
2136 return (arg: string): func(string): string => ((sep: string): string => {
2137 return sep .. arg
2138 })
2139 enddef
2140 assert_equal('--there', F('unused')('there')('--'))
2141 END
2142 CheckScriptSuccess(lines)
2143enddef
2144
Bram Moolenaar52bf81c2020-11-17 18:50:44 +01002145def Shadowed(): list<number>
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01002146 var FuncList: list<func: number> = [() => 42]
Bram Moolenaar75ab91f2021-01-10 22:42:50 +01002147 return FuncList->mapnew((_, Shadowed) => Shadowed())
Bram Moolenaar52bf81c2020-11-17 18:50:44 +01002148enddef
2149
2150def Test_lambda_arg_shadows_func()
2151 assert_equal([42], Shadowed())
2152enddef
2153
Bram Moolenaaracd4c5e2020-06-22 19:39:03 +02002154def Line_continuation_in_def(dir: string = ''): string
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002155 var path: string = empty(dir)
2156 \ ? 'empty'
2157 \ : 'full'
2158 return path
Bram Moolenaaracd4c5e2020-06-22 19:39:03 +02002159enddef
2160
2161def Test_line_continuation_in_def()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002162 Line_continuation_in_def('.')->assert_equal('full')
Bram Moolenaaracd4c5e2020-06-22 19:39:03 +02002163enddef
2164
Bram Moolenaar2ea95b62020-11-19 21:47:56 +01002165def Test_script_var_in_lambda()
2166 var lines =<< trim END
2167 vim9script
2168 var script = 'test'
Bram Moolenaarbb8a7ce2021-04-10 20:10:26 +02002169 assert_equal(['test'], map(['one'], (_, _) => script))
Bram Moolenaar2ea95b62020-11-19 21:47:56 +01002170 END
2171 CheckScriptSuccess(lines)
2172enddef
2173
Bram Moolenaar5e654232020-09-16 15:22:00 +02002174def Line_continuation_in_lambda(): list<string>
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002175 var x = range(97, 100)
Bram Moolenaar75ab91f2021-01-10 22:42:50 +01002176 ->mapnew((_, v) => nr2char(v)
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01002177 ->toupper())
Bram Moolenaar7a4b8982020-07-08 17:36:21 +02002178 ->reverse()
2179 return x
2180enddef
2181
2182def Test_line_continuation_in_lambda()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002183 Line_continuation_in_lambda()->assert_equal(['D', 'C', 'B', 'A'])
Bram Moolenaarf898f7c2021-01-16 18:09:52 +01002184
2185 var lines =<< trim END
2186 vim9script
2187 var res = [{n: 1, m: 2, s: 'xxx'}]
2188 ->mapnew((_, v: dict<any>): string => printf('%d:%d:%s',
2189 v.n,
2190 v.m,
2191 substitute(v.s, '.*', 'yyy', '')
2192 ))
2193 assert_equal(['1:2:yyy'], res)
2194 END
2195 CheckScriptSuccess(lines)
Bram Moolenaar7a4b8982020-07-08 17:36:21 +02002196enddef
2197
Bram Moolenaarb6571982021-01-08 22:24:19 +01002198def Test_list_lambda()
2199 timer_start(1000, (_) => 0)
2200 var body = execute(timer_info()[0].callback
2201 ->string()
2202 ->substitute("('", ' ', '')
2203 ->substitute("')", '', '')
2204 ->substitute('function\zs', ' ', ''))
Bram Moolenaar767034c2021-04-09 17:24:52 +02002205 assert_match('def <lambda>\d\+(_: any): number\n1 return 0\n enddef', body)
Bram Moolenaarb6571982021-01-08 22:24:19 +01002206enddef
2207
Bram Moolenaar96cf4ba2021-04-24 14:15:41 +02002208def Test_legacy_lambda()
2209 legacy echo {x -> 'hello ' .. x}('foo')
Bram Moolenaardc4c2302021-04-25 13:54:42 +02002210
Bram Moolenaar96cf4ba2021-04-24 14:15:41 +02002211 var lines =<< trim END
2212 echo {x -> 'hello ' .. x}('foo')
2213 END
2214 CheckDefAndScriptFailure(lines, 'E720:')
Bram Moolenaardc4c2302021-04-25 13:54:42 +02002215
2216 lines =<< trim END
2217 vim9script
2218 def Func()
2219 echo (() => 'no error')()
2220 enddef
2221 legacy call s:Func()
2222 END
2223 CheckScriptSuccess(lines)
Bram Moolenaar96cf4ba2021-04-24 14:15:41 +02002224enddef
2225
Bram Moolenaarc3cb1c92021-06-02 16:47:53 +02002226def Test_legacy_errors()
2227 for cmd in ['if', 'elseif', 'else', 'endif',
2228 'for', 'endfor', 'continue', 'break',
2229 'while', 'endwhile',
2230 'try', 'catch', 'finally', 'endtry']
2231 CheckDefFailure(['legacy ' .. cmd .. ' expr'], 'E1189:')
2232 endfor
2233enddef
2234
Bram Moolenaarab360522021-01-10 14:02:28 +01002235def DoFilterThis(a: string): list<string>
2236 # closure nested inside another closure using argument
2237 var Filter = (l) => filter(l, (_, v) => stridx(v, a) == 0)
2238 return ['x', 'y', 'a', 'x2', 'c']->Filter()
2239enddef
2240
2241def Test_nested_closure_using_argument()
2242 assert_equal(['x', 'x2'], DoFilterThis('x'))
2243enddef
2244
Bram Moolenaar0186e582021-01-10 18:33:11 +01002245def Test_triple_nested_closure()
2246 var what = 'x'
2247 var Match = (val: string, cmp: string): bool => stridx(val, cmp) == 0
2248 var Filter = (l) => filter(l, (_, v) => Match(v, what))
2249 assert_equal(['x', 'x2'], ['x', 'y', 'a', 'x2', 'c']->Filter())
2250enddef
2251
Bram Moolenaar8f510af2020-07-05 18:48:23 +02002252func Test_silent_echo()
Bram Moolenaar47e7d702020-07-05 18:18:42 +02002253 CheckScreendump
2254
2255 let lines =<< trim END
2256 vim9script
2257 def EchoNothing()
2258 silent echo ''
2259 enddef
2260 defcompile
2261 END
Bram Moolenaar8f510af2020-07-05 18:48:23 +02002262 call writefile(lines, 'XTest_silent_echo')
Bram Moolenaar47e7d702020-07-05 18:18:42 +02002263
2264 " Check that the balloon shows up after a mouse move
2265 let buf = RunVimInTerminal('-S XTest_silent_echo', {'rows': 6})
Bram Moolenaar8f510af2020-07-05 18:48:23 +02002266 call term_sendkeys(buf, ":abc")
Bram Moolenaar47e7d702020-07-05 18:18:42 +02002267 call VerifyScreenDump(buf, 'Test_vim9_silent_echo', {})
2268
2269 " clean up
2270 call StopVimInTerminal(buf)
2271 call delete('XTest_silent_echo')
Bram Moolenaar8f510af2020-07-05 18:48:23 +02002272endfunc
Bram Moolenaar47e7d702020-07-05 18:18:42 +02002273
Bram Moolenaar171fb922020-10-28 16:54:47 +01002274def SilentlyError()
2275 execute('silent! invalid')
2276 g:did_it = 'yes'
2277enddef
2278
Bram Moolenaar28ee8922020-10-28 20:20:00 +01002279func UserError()
2280 silent! invalid
2281endfunc
2282
2283def SilentlyUserError()
2284 UserError()
2285 g:did_it = 'yes'
2286enddef
Bram Moolenaar171fb922020-10-28 16:54:47 +01002287
2288" This can't be a :def function, because the assert would not be reached.
Bram Moolenaar171fb922020-10-28 16:54:47 +01002289func Test_ignore_silent_error()
2290 let g:did_it = 'no'
2291 call SilentlyError()
2292 call assert_equal('yes', g:did_it)
2293
Bram Moolenaar28ee8922020-10-28 20:20:00 +01002294 let g:did_it = 'no'
2295 call SilentlyUserError()
2296 call assert_equal('yes', g:did_it)
Bram Moolenaar171fb922020-10-28 16:54:47 +01002297
2298 unlet g:did_it
2299endfunc
2300
Bram Moolenaarcd030c42020-10-30 21:49:40 +01002301def Test_ignore_silent_error_in_filter()
2302 var lines =<< trim END
2303 vim9script
2304 def Filter(winid: number, key: string): bool
2305 if key == 'o'
2306 silent! eval [][0]
2307 return true
2308 endif
2309 return popup_filter_menu(winid, key)
2310 enddef
2311
Bram Moolenaare0de1712020-12-02 17:36:54 +01002312 popup_create('popup', {filter: Filter})
Bram Moolenaarcd030c42020-10-30 21:49:40 +01002313 feedkeys("o\r", 'xnt')
2314 END
2315 CheckScriptSuccess(lines)
2316enddef
2317
Bram Moolenaar4b9bd692020-09-05 21:57:53 +02002318def Fibonacci(n: number): number
2319 if n < 2
2320 return n
2321 else
2322 return Fibonacci(n - 1) + Fibonacci(n - 2)
2323 endif
2324enddef
2325
Bram Moolenaar985116a2020-07-12 17:31:09 +02002326def Test_recursive_call()
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002327 Fibonacci(20)->assert_equal(6765)
Bram Moolenaar985116a2020-07-12 17:31:09 +02002328enddef
2329
Bram Moolenaar08f7a412020-07-13 20:41:08 +02002330def TreeWalk(dir: string): list<any>
Bram Moolenaar75ab91f2021-01-10 22:42:50 +01002331 return readdir(dir)->mapnew((_, val) =>
Bram Moolenaar08f7a412020-07-13 20:41:08 +02002332 fnamemodify(dir .. '/' .. val, ':p')->isdirectory()
Bram Moolenaar2bede172020-11-19 18:53:18 +01002333 ? {[val]: TreeWalk(dir .. '/' .. val)}
Bram Moolenaar08f7a412020-07-13 20:41:08 +02002334 : val
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01002335 )
Bram Moolenaar08f7a412020-07-13 20:41:08 +02002336enddef
2337
2338def Test_closure_in_map()
2339 mkdir('XclosureDir/tdir', 'p')
2340 writefile(['111'], 'XclosureDir/file1')
2341 writefile(['222'], 'XclosureDir/file2')
2342 writefile(['333'], 'XclosureDir/tdir/file3')
2343
Bram Moolenaare0de1712020-12-02 17:36:54 +01002344 TreeWalk('XclosureDir')->assert_equal(['file1', 'file2', {tdir: ['file3']}])
Bram Moolenaar08f7a412020-07-13 20:41:08 +02002345
2346 delete('XclosureDir', 'rf')
2347enddef
2348
Bram Moolenaar7b5d5442020-10-04 13:42:34 +02002349def Test_invalid_function_name()
2350 var lines =<< trim END
2351 vim9script
2352 def s: list<string>
2353 END
2354 CheckScriptFailure(lines, 'E129:')
2355
2356 lines =<< trim END
2357 vim9script
2358 def g: list<string>
2359 END
2360 CheckScriptFailure(lines, 'E129:')
2361
2362 lines =<< trim END
2363 vim9script
2364 def <SID>: list<string>
2365 END
2366 CheckScriptFailure(lines, 'E884:')
2367
2368 lines =<< trim END
2369 vim9script
2370 def F list<string>
2371 END
2372 CheckScriptFailure(lines, 'E488:')
2373enddef
2374
Bram Moolenaara90afb92020-07-15 22:38:56 +02002375def Test_partial_call()
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002376 var Xsetlist = function('setloclist', [0])
Bram Moolenaare0de1712020-12-02 17:36:54 +01002377 Xsetlist([], ' ', {title: 'test'})
2378 getloclist(0, {title: 1})->assert_equal({title: 'test'})
Bram Moolenaara90afb92020-07-15 22:38:56 +02002379
2380 Xsetlist = function('setloclist', [0, [], ' '])
Bram Moolenaare0de1712020-12-02 17:36:54 +01002381 Xsetlist({title: 'test'})
2382 getloclist(0, {title: 1})->assert_equal({title: 'test'})
Bram Moolenaara90afb92020-07-15 22:38:56 +02002383
2384 Xsetlist = function('setqflist')
Bram Moolenaare0de1712020-12-02 17:36:54 +01002385 Xsetlist([], ' ', {title: 'test'})
2386 getqflist({title: 1})->assert_equal({title: 'test'})
Bram Moolenaara90afb92020-07-15 22:38:56 +02002387
2388 Xsetlist = function('setqflist', [[], ' '])
Bram Moolenaare0de1712020-12-02 17:36:54 +01002389 Xsetlist({title: 'test'})
2390 getqflist({title: 1})->assert_equal({title: 'test'})
Bram Moolenaar6abd3dc2020-10-04 14:17:32 +02002391
2392 var Len: func: number = function('len', ['word'])
2393 assert_equal(4, Len())
Bram Moolenaara90afb92020-07-15 22:38:56 +02002394enddef
2395
Bram Moolenaar2dd0a2c2020-08-08 15:10:27 +02002396def Test_cmd_modifier()
2397 tab echo '0'
Bram Moolenaard2c61702020-09-06 15:58:36 +02002398 CheckDefFailure(['5tab echo 3'], 'E16:')
Bram Moolenaar2dd0a2c2020-08-08 15:10:27 +02002399enddef
2400
2401def Test_restore_modifiers()
2402 # check that when compiling a :def function command modifiers are not messed
2403 # up.
Bram Moolenaar7a9cbca2020-09-27 22:47:05 +02002404 var lines =<< trim END
Bram Moolenaar2dd0a2c2020-08-08 15:10:27 +02002405 vim9script
2406 set eventignore=
2407 autocmd QuickFixCmdPost * copen
2408 def AutocmdsDisabled()
Bram Moolenaar6cf7e3b2020-10-28 14:31:16 +01002409 eval 0
Bram Moolenaar2dd0a2c2020-08-08 15:10:27 +02002410 enddef
2411 func Func()
2412 noautocmd call s:AutocmdsDisabled()
2413 let g:ei_after = &eventignore
2414 endfunc
2415 Func()
2416 END
2417 CheckScriptSuccess(lines)
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002418 g:ei_after->assert_equal('')
Bram Moolenaar2dd0a2c2020-08-08 15:10:27 +02002419enddef
2420
Bram Moolenaardfa3d552020-09-10 22:05:08 +02002421def StackTop()
2422 eval 1
2423 eval 2
2424 # call not on fourth line
2425 StackBot()
2426enddef
2427
2428def StackBot()
2429 # throw an error
2430 eval [][0]
2431enddef
2432
2433def Test_callstack_def()
2434 try
2435 StackTop()
2436 catch
Bram Moolenaarc0c71e92020-09-11 19:09:48 +02002437 v:throwpoint->assert_match('Test_callstack_def\[2\]..StackTop\[4\]..StackBot, line 2')
Bram Moolenaardfa3d552020-09-10 22:05:08 +02002438 endtry
2439enddef
2440
Bram Moolenaare8211a32020-10-09 22:04:29 +02002441" Re-using spot for variable used in block
2442def Test_block_scoped_var()
2443 var lines =<< trim END
2444 vim9script
2445 def Func()
2446 var x = ['a', 'b', 'c']
2447 if 1
2448 var y = 'x'
Bram Moolenaarbb8a7ce2021-04-10 20:10:26 +02002449 map(x, (_, _) => y)
Bram Moolenaare8211a32020-10-09 22:04:29 +02002450 endif
2451 var z = x
2452 assert_equal(['x', 'x', 'x'], z)
2453 enddef
2454 Func()
2455 END
2456 CheckScriptSuccess(lines)
2457enddef
2458
Bram Moolenaareeece9e2020-11-20 19:26:48 +01002459def Test_reset_did_emsg()
2460 var lines =<< trim END
2461 @s = 'blah'
2462 au BufWinLeave * #
2463 def Func()
2464 var winid = popup_create('popup', {})
2465 exe '*s'
2466 popup_close(winid)
2467 enddef
2468 Func()
2469 END
2470 CheckScriptFailure(lines, 'E492:', 8)
Bram Moolenaar2d870f82020-12-05 13:41:01 +01002471 delfunc! g:Func
Bram Moolenaareeece9e2020-11-20 19:26:48 +01002472enddef
2473
Bram Moolenaar57f799e2020-12-12 20:42:19 +01002474def Test_did_emsg_reset()
2475 # executing an autocommand resets did_emsg, this should not result in a
2476 # builtin function considered failing
2477 var lines =<< trim END
2478 vim9script
2479 au BufWinLeave * #
2480 def Func()
Bram Moolenaar767034c2021-04-09 17:24:52 +02002481 popup_menu('', {callback: (a, b) => popup_create('', {})->popup_close()})
Bram Moolenaar57f799e2020-12-12 20:42:19 +01002482 eval [][0]
2483 enddef
2484 nno <F3> <cmd>call <sid>Func()<cr>
2485 feedkeys("\<F3>\e", 'xt')
2486 END
2487 writefile(lines, 'XemsgReset')
2488 assert_fails('so XemsgReset', ['E684:', 'E684:'], lines, 2)
2489 delete('XemsgReset')
2490 nunmap <F3>
2491 au! BufWinLeave
2492enddef
2493
Bram Moolenaar56602ba2020-12-05 21:22:08 +01002494def Test_abort_with_silent_call()
2495 var lines =<< trim END
2496 vim9script
2497 g:result = 'none'
2498 def Func()
2499 g:result += 3
2500 g:result = 'yes'
2501 enddef
2502 # error is silenced, but function aborts on error
2503 silent! Func()
2504 assert_equal('none', g:result)
2505 unlet g:result
2506 END
2507 CheckScriptSuccess(lines)
2508enddef
2509
Bram Moolenaarf665e972020-12-05 19:17:16 +01002510def Test_continues_with_silent_error()
2511 var lines =<< trim END
2512 vim9script
2513 g:result = 'none'
2514 def Func()
2515 silent! g:result += 3
2516 g:result = 'yes'
2517 enddef
2518 # error is silenced, function does not abort
2519 Func()
2520 assert_equal('yes', g:result)
2521 unlet g:result
2522 END
2523 CheckScriptSuccess(lines)
2524enddef
2525
Bram Moolenaaraf0df472020-12-02 20:51:22 +01002526def Test_abort_even_with_silent()
2527 var lines =<< trim END
2528 vim9script
2529 g:result = 'none'
2530 def Func()
2531 eval {-> ''}() .. '' .. {}['X']
2532 g:result = 'yes'
2533 enddef
Bram Moolenaarf665e972020-12-05 19:17:16 +01002534 silent! Func()
Bram Moolenaaraf0df472020-12-02 20:51:22 +01002535 assert_equal('none', g:result)
Bram Moolenaar4029cab2020-12-05 18:13:27 +01002536 unlet g:result
2537 END
2538 CheckScriptSuccess(lines)
2539enddef
2540
Bram Moolenaarf665e972020-12-05 19:17:16 +01002541def Test_cmdmod_silent_restored()
2542 var lines =<< trim END
2543 vim9script
2544 def Func()
2545 g:result = 'none'
2546 silent! g:result += 3
2547 g:result = 'none'
2548 g:result += 3
2549 enddef
2550 Func()
2551 END
2552 # can't use CheckScriptFailure, it ignores the :silent!
2553 var fname = 'Xdefsilent'
2554 writefile(lines, fname)
2555 var caught = 'no'
2556 try
2557 exe 'source ' .. fname
2558 catch /E1030:/
2559 caught = 'yes'
2560 assert_match('Func, line 4', v:throwpoint)
2561 endtry
2562 assert_equal('yes', caught)
2563 delete(fname)
2564enddef
2565
Bram Moolenaar2fecb532021-03-24 22:00:56 +01002566def Test_cmdmod_silent_nested()
2567 var lines =<< trim END
2568 vim9script
2569 var result = ''
2570
2571 def Error()
2572 result ..= 'Eb'
2573 eval [][0]
2574 result ..= 'Ea'
2575 enddef
2576
2577 def Crash()
2578 result ..= 'Cb'
2579 sil! Error()
2580 result ..= 'Ca'
2581 enddef
2582
2583 Crash()
2584 assert_equal('CbEbEaCa', result)
2585 END
2586 CheckScriptSuccess(lines)
2587enddef
2588
Bram Moolenaar4029cab2020-12-05 18:13:27 +01002589def Test_dict_member_with_silent()
2590 var lines =<< trim END
2591 vim9script
2592 g:result = 'none'
2593 var d: dict<any>
2594 def Func()
2595 try
Bram Moolenaar2949cfd2020-12-31 21:28:47 +01002596 g:result = map([], (_, v) => ({}[v]))->join() .. d['']
Bram Moolenaar4029cab2020-12-05 18:13:27 +01002597 catch
2598 endtry
2599 enddef
2600 silent! Func()
2601 assert_equal('0', g:result)
2602 unlet g:result
Bram Moolenaaraf0df472020-12-02 20:51:22 +01002603 END
2604 CheckScriptSuccess(lines)
2605enddef
2606
Bram Moolenaarf9041332021-01-21 19:41:16 +01002607def Test_skip_cmds_with_silent()
2608 var lines =<< trim END
2609 vim9script
2610
2611 def Func(b: bool)
2612 Crash()
2613 enddef
2614
2615 def Crash()
2616 sil! :/not found/d _
2617 sil! :/not found/put _
2618 enddef
2619
2620 Func(true)
2621 END
2622 CheckScriptSuccess(lines)
2623enddef
2624
Bram Moolenaar5b3d1bb2020-12-22 12:20:08 +01002625def Test_opfunc()
2626 nnoremap <F3> <cmd>set opfunc=Opfunc<cr>g@
2627 def g:Opfunc(_: any): string
2628 setline(1, 'ASDF')
2629 return ''
2630 enddef
2631 new
2632 setline(1, 'asdf')
2633 feedkeys("\<F3>$", 'x')
2634 assert_equal('ASDF', getline(1))
2635
2636 bwipe!
2637 nunmap <F3>
2638enddef
2639
Bram Moolenaar077a4232020-12-22 18:33:27 +01002640" this was crashing on exit
2641def Test_nested_lambda_in_closure()
2642 var lines =<< trim END
2643 vim9script
Bram Moolenaar227c58a2021-04-28 20:40:44 +02002644 command WriteDone writefile(['Done'], 'XnestedDone')
Bram Moolenaar077a4232020-12-22 18:33:27 +01002645 def Outer()
2646 def g:Inner()
2647 echo map([1, 2, 3], {_, v -> v + 1})
2648 enddef
2649 g:Inner()
2650 enddef
2651 defcompile
Bram Moolenaar227c58a2021-04-28 20:40:44 +02002652 # not reached
Bram Moolenaar077a4232020-12-22 18:33:27 +01002653 END
Bram Moolenaar227c58a2021-04-28 20:40:44 +02002654 if !RunVim([], lines, '--clean -c WriteDone -c quit')
Bram Moolenaar077a4232020-12-22 18:33:27 +01002655 return
2656 endif
2657 assert_equal(['Done'], readfile('XnestedDone'))
2658 delete('XnestedDone')
2659enddef
2660
Bram Moolenaar04947cc2021-03-06 19:26:46 +01002661def Test_check_func_arg_types()
2662 var lines =<< trim END
2663 vim9script
2664 def F1(x: string): string
2665 return x
2666 enddef
2667
2668 def F2(x: number): number
2669 return x + 1
2670 enddef
2671
2672 def G(g: func): dict<func>
2673 return {f: g}
2674 enddef
2675
2676 def H(d: dict<func>): string
2677 return d.f('a')
2678 enddef
2679 END
2680
2681 CheckScriptSuccess(lines + ['echo H(G(F1))'])
2682 CheckScriptFailure(lines + ['echo H(G(F2))'], 'E1013:')
2683enddef
2684
Bram Moolenaar701cc6c2021-04-10 13:33:48 +02002685def Test_compile_error()
2686 var lines =<< trim END
2687 def g:Broken()
2688 echo 'a' + {}
2689 enddef
2690 call g:Broken()
2691 END
2692 # First call: compilation error
2693 CheckScriptFailure(lines, 'E1051: Wrong argument type for +')
2694
2695 # Second call won't try compiling again
2696 assert_fails('call g:Broken()', 'E1091: Function is not compiled: Broken')
Bram Moolenaar599410c2021-04-10 14:03:43 +02002697 delfunc g:Broken
2698
2699 # No error when compiling with :silent!
2700 lines =<< trim END
2701 def g:Broken()
2702 echo 'a' + []
2703 enddef
2704 silent! defcompile
2705 END
2706 CheckScriptSuccess(lines)
2707
2708 # Calling the function won't try compiling again
2709 assert_fails('call g:Broken()', 'E1091: Function is not compiled: Broken')
2710 delfunc g:Broken
Bram Moolenaar701cc6c2021-04-10 13:33:48 +02002711enddef
2712
Bram Moolenaar962c43b2021-04-10 17:18:09 +02002713def Test_ignored_argument()
2714 var lines =<< trim END
2715 vim9script
2716 def Ignore(_, _): string
2717 return 'yes'
2718 enddef
2719 assert_equal('yes', Ignore(1, 2))
2720
2721 func Ok(_)
2722 return a:_
2723 endfunc
2724 assert_equal('ok', Ok('ok'))
2725
2726 func Oktoo()
2727 let _ = 'too'
2728 return _
2729 endfunc
2730 assert_equal('too', Oktoo())
Bram Moolenaarda479c72021-04-10 21:01:38 +02002731
2732 assert_equal([[1], [2], [3]], range(3)->mapnew((_, v) => [v]->map((_, w) => w + 1)))
Bram Moolenaar962c43b2021-04-10 17:18:09 +02002733 END
2734 CheckScriptSuccess(lines)
2735
2736 lines =<< trim END
2737 def Ignore(_: string): string
2738 return _
2739 enddef
2740 defcompile
2741 END
2742 CheckScriptFailure(lines, 'E1181:', 1)
2743
2744 lines =<< trim END
2745 var _ = 1
2746 END
2747 CheckDefAndScriptFailure(lines, 'E1181:', 1)
Yegappan Lakshmanan34fcb692021-05-25 20:14:00 +02002748
2749 lines =<< trim END
2750 var x = _
2751 END
2752 CheckDefAndScriptFailure(lines, 'E1181:', 1)
Bram Moolenaar962c43b2021-04-10 17:18:09 +02002753enddef
2754
Bram Moolenaarbb8a7ce2021-04-10 20:10:26 +02002755def Test_too_many_arguments()
2756 var lines =<< trim END
2757 echo [0, 1, 2]->map(() => 123)
2758 END
2759 CheckDefExecAndScriptFailure(lines, 'E1106: 2 arguments too many', 1)
2760
2761 lines =<< trim END
2762 echo [0, 1, 2]->map((_) => 123)
2763 END
2764 CheckDefExecAndScriptFailure(lines, 'E1106: One argument too many', 1)
2765enddef
Bram Moolenaar077a4232020-12-22 18:33:27 +01002766
Bram Moolenaara6aa1642021-04-23 19:32:23 +02002767def Test_closing_brace_at_start_of_line()
2768 var lines =<< trim END
2769 def Func()
2770 enddef
2771 Func(
2772 )
2773 END
2774 call CheckDefAndScriptSuccess(lines)
2775enddef
2776
Bram Moolenaar20677332021-06-06 17:02:53 +02002777if has('python3')
2778 def Test_python3_heredoc()
2779 py3 << trim EOF
2780 import vim
2781 vim.vars['didit'] = 'yes'
2782 EOF
2783 assert_equal('yes', g:didit)
2784
2785 python3 << trim EOF
2786 import vim
2787 vim.vars['didit'] = 'again'
2788 EOF
2789 assert_equal('again', g:didit)
2790 enddef
2791endif
2792
2793" This messes up syntax highlight, keep near the end.
2794if has('lua')
2795 def Test_lua_heredoc()
2796 g:d = {}
2797 lua << trim EOF
2798 x = vim.eval('g:d')
2799 x['key'] = 'val'
2800 EOF
2801 assert_equal('val', g:d.key)
2802 enddef
2803endif
2804
Bram Moolenaarf7779c62020-05-03 15:38:16 +02002805
Bram Moolenaar5deeb3f2020-04-05 17:08:17 +02002806" vim: ts=8 sw=2 sts=2 expandtab tw=80 fdm=marker