blob: 3608802b1902bdd4a6541f0acb554e0f266bc992 [file] [log] [blame]
Bram Moolenaard8448622022-01-07 21:39:52 +00001" Test import/export of the Vim9 script language.
Bram Moolenaar160aa862022-01-10 21:29:57 +00002" Also the autoload mechanism.
Bram Moolenaard8448622022-01-07 21:39:52 +00003
4source check.vim
5source term_util.vim
6source vim9.vim
7
8let s:export_script_lines =<< trim END
9 vim9script
10 var name: string = 'bob'
11 def Concat(arg: string): string
12 return name .. arg
13 enddef
14 g:result = Concat('bie')
15 g:localname = name
16
17 export const CONST = 1234
18 export var exported = 9876
19 export var exp_name = 'John'
20 export def Exported(): string
21 return 'Exported'
22 enddef
23 export def ExportedValue(): number
24 return exported
25 enddef
26 export def ExportedInc()
27 exported += 5
28 enddef
29 export final theList = [1]
Bram Moolenaar857c8bb2022-01-15 21:08:19 +000030 export def AddSome(s: string): string
31 return s .. 'some'
32 enddef
33 export var AddRef = AddSome
Bram Moolenaard8448622022-01-07 21:39:52 +000034END
35
36def Undo_export_script_lines()
37 unlet g:result
38 unlet g:localname
39enddef
40
41def Test_vim9_import_export()
42 writefile(s:export_script_lines, 'Xexport.vim')
43 var import_script_lines =<< trim END
44 vim9script
45 var dir = './'
46 var ext = ".vim"
47 import dir .. 'Xexport' .. ext as expo
48
49 g:exported1 = expo.exported
50 expo.exported += 3
51 g:exported2 = expo.exported
52 g:exported3 = expo.ExportedValue()
53
54 expo.ExportedInc()
55 g:exported_i1 = expo.exported
56 g:exported_i2 = expo.ExportedValue()
57
58 expo.exported = 11
59 g:exported_s1 = expo.exported
60 g:exported_s2 = expo.ExportedValue()
61
62 g:imported_func = expo.Exported()
63
64 def GetExported(): string
65 var local_dict = {ref: expo.Exported}
66 return local_dict.ref()
67 enddef
68 g:funcref_result = GetExported()
69
Bram Moolenaar21f0d6c2022-01-20 17:35:49 +000070 def GetName(): string
71 return expo.exp_name .. 'son'
72 enddef
73 g:long_name = GetName()
74
Bram Moolenaard8448622022-01-07 21:39:52 +000075 g:imported_name = expo.exp_name
76 expo.exp_name ..= ' Doe'
Bram Moolenaar47036b62022-01-16 21:18:53 +000077 expo.exp_name = expo.exp_name .. ' Maar'
Bram Moolenaard8448622022-01-07 21:39:52 +000078 g:imported_name_appended = expo.exp_name
79 g:exported_later = expo.exported
80
81 expo.theList->add(2)
82 assert_equal([1, 2], expo.theList)
Bram Moolenaar857c8bb2022-01-15 21:08:19 +000083
84 assert_equal('andthensome', 'andthen'->expo.AddSome())
85 assert_equal('awesome', 'awe'->expo.AddRef())
Bram Moolenaard8448622022-01-07 21:39:52 +000086 END
87 writefile(import_script_lines, 'Ximport.vim')
88 source Ximport.vim
89
90 assert_equal('bobbie', g:result)
91 assert_equal('bob', g:localname)
92 assert_equal(9876, g:exported1)
93 assert_equal(9879, g:exported2)
94 assert_equal(9879, g:exported3)
95
96 assert_equal(9884, g:exported_i1)
97 assert_equal(9884, g:exported_i2)
98
99 assert_equal(11, g:exported_s1)
100 assert_equal(11, g:exported_s2)
101 assert_equal(11, g:exported_later)
102
103 assert_equal('Exported', g:imported_func)
104 assert_equal('Exported', g:funcref_result)
105 assert_equal('John', g:imported_name)
Bram Moolenaar21f0d6c2022-01-20 17:35:49 +0000106 assert_equal('Johnson', g:long_name)
Bram Moolenaar47036b62022-01-16 21:18:53 +0000107 assert_equal('John Doe Maar', g:imported_name_appended)
Bram Moolenaard8448622022-01-07 21:39:52 +0000108 assert_false(exists('g:name'))
109
110 Undo_export_script_lines()
111 unlet g:exported1
112 unlet g:exported2
113 unlet g:exported3
114 unlet g:exported_i1
115 unlet g:exported_i2
116 unlet g:exported_later
117 unlet g:imported_func
Bram Moolenaar21f0d6c2022-01-20 17:35:49 +0000118 unlet g:imported_name g:long_name g:imported_name_appended
Bram Moolenaard8448622022-01-07 21:39:52 +0000119 delete('Ximport.vim')
120
121 # similar, with line breaks
122 var import_line_break_script_lines =<< trim END
123 vim9script
124 import './Xexport.vim'
125 as expo
126 g:exported = expo.exported
127 expo.exported += 7
128 g:exported_added = expo.exported
129 g:imported_func = expo.Exported()
130 END
131 writefile(import_line_break_script_lines, 'Ximport_lbr.vim')
132 source Ximport_lbr.vim
133
134 assert_equal(11, g:exported)
135 assert_equal(18, g:exported_added)
136 assert_equal('Exported', g:imported_func)
137
138 # exported script not sourced again
139 assert_false(exists('g:result'))
140 unlet g:exported
141 unlet g:exported_added
142 unlet g:imported_func
143 delete('Ximport_lbr.vim')
144
145 var line_break_before_dot =<< trim END
146 vim9script
147 import './Xexport.vim' as expo
148 g:exported = expo
149 .exported
150 END
151 writefile(line_break_before_dot, 'Ximport_lbr_before_dot.vim')
152 assert_fails('source Ximport_lbr_before_dot.vim', 'E1060:', '', 3)
153 delete('Ximport_lbr_before_dot.vim')
154
155 var line_break_after_dot =<< trim END
156 vim9script
157 import './Xexport.vim' as expo
158 g:exported = expo.
159 exported
160 END
161 writefile(line_break_after_dot, 'Ximport_lbr_after_dot.vim')
162 assert_fails('source Ximport_lbr_after_dot.vim', 'E1074:', '', 3)
163 delete('Ximport_lbr_after_dot.vim')
164
165 var import_star_as_lines =<< trim END
166 vim9script
167 import './Xexport.vim' as Export
168 def UseExport()
169 g:exported_def = Export.exported
170 enddef
171 g:exported_script = Export.exported
172 assert_equal(1, exists('Export.exported'))
173 assert_equal(0, exists('Export.notexported'))
174 UseExport()
175 END
176 writefile(import_star_as_lines, 'Ximport.vim')
177 source Ximport.vim
178
179 assert_equal(18, g:exported_def)
180 assert_equal(18, g:exported_script)
181 unlet g:exported_def
182 unlet g:exported_script
183
184 var import_star_as_lines_no_dot =<< trim END
185 vim9script
186 import './Xexport.vim' as Export
187 def Func()
188 var dummy = 1
189 var imported = Export + dummy
190 enddef
191 defcompile
192 END
193 writefile(import_star_as_lines_no_dot, 'Ximport.vim')
194 assert_fails('source Ximport.vim', 'E1060:', '', 2, 'Func')
195
196 var import_star_as_lines_dot_space =<< trim END
197 vim9script
198 import './Xexport.vim' as Export
199 def Func()
200 var imported = Export . exported
201 enddef
202 defcompile
203 END
204 writefile(import_star_as_lines_dot_space, 'Ximport.vim')
205 assert_fails('source Ximport.vim', 'E1074:', '', 1, 'Func')
206
Bram Moolenaar7c24dfd2022-01-08 17:03:55 +0000207 writefile(s:export_script_lines, 'Xexport2.vim')
208 var import_as_duplicated =<< trim END
Bram Moolenaard8448622022-01-07 21:39:52 +0000209 vim9script
210 import './Xexport.vim' as expo
Bram Moolenaar7c24dfd2022-01-08 17:03:55 +0000211 import './Xexport2.vim' as expo
Bram Moolenaard8448622022-01-07 21:39:52 +0000212 END
Bram Moolenaar7c24dfd2022-01-08 17:03:55 +0000213 writefile(import_as_duplicated, 'Ximport.vim')
Bram Moolenaard8448622022-01-07 21:39:52 +0000214 assert_fails('source Ximport.vim', 'E1073:', '', 3, 'Ximport.vim')
Bram Moolenaar7c24dfd2022-01-08 17:03:55 +0000215 delete('Xexport2.vim')
Bram Moolenaard8448622022-01-07 21:39:52 +0000216
217 var import_star_as_lines_script_no_dot =<< trim END
218 vim9script
219 import './Xexport.vim' as Export
220 g:imported_script = Export exported
221 END
222 writefile(import_star_as_lines_script_no_dot, 'Ximport.vim')
223 assert_fails('source Ximport.vim', 'E1060: Expected dot after name: Export exported')
224
225 var import_star_as_lines_script_space_after_dot =<< trim END
226 vim9script
227 import './Xexport.vim' as Export
228 g:imported_script = Export. exported
229 END
230 writefile(import_star_as_lines_script_space_after_dot, 'Ximport.vim')
231 assert_fails('source Ximport.vim', 'E1074:')
232
233 var import_star_as_lines_missing_name =<< trim END
234 vim9script
235 import './Xexport.vim' as Export
236 def Func()
237 var imported = Export.
238 enddef
239 defcompile
240 END
241 writefile(import_star_as_lines_missing_name, 'Ximport.vim')
242 assert_fails('source Ximport.vim', 'E1048:', '', 1, 'Func')
243
244 var import_star_as_lbr_lines =<< trim END
245 vim9script
246 import './Xexport.vim'
247 as Export
248 def UseExport()
249 g:exported = Export.exported
250 enddef
251 UseExport()
252 END
253 writefile(import_star_as_lbr_lines, 'Ximport.vim')
254 source Ximport.vim
255 assert_equal(18, g:exported)
256 unlet g:exported
257
258 # try to use something that exists but is not exported
259 var import_not_exported_lines =<< trim END
260 vim9script
261 import './Xexport.vim' as expo
262 echo expo.name
263 END
264 writefile(import_not_exported_lines, 'Ximport.vim')
265 assert_fails('source Ximport.vim', 'E1049:', '', 3, 'Ximport.vim')
266
267 # try to import something that is already defined
268 var import_already_defined =<< trim END
269 vim9script
270 var exported = 'something'
271 import './Xexport.vim' as exported
272 END
273 writefile(import_already_defined, 'Ximport.vim')
274 assert_fails('source Ximport.vim', 'E1054:', '', 3, 'Ximport.vim')
275
276 # try changing an imported const
277 var import_assign_to_const =<< trim END
278 vim9script
279 import './Xexport.vim' as expo
280 def Assign()
281 expo.CONST = 987
282 enddef
283 defcompile
284 END
285 writefile(import_assign_to_const, 'Ximport.vim')
286 assert_fails('source Ximport.vim', 'E46:', '', 1, '_Assign')
287
288 # try changing an imported final
289 var import_assign_to_final =<< trim END
290 vim9script
291 import './Xexport.vim' as expo
292 def Assign()
293 expo.theList = [2]
294 enddef
295 defcompile
296 END
297 writefile(import_assign_to_final, 'Ximport.vim')
298 assert_fails('source Ximport.vim', 'E46:', '', 1, '_Assign')
299
300 var import_no_as_lines =<< trim END
301 vim9script
302 import './Xexport.vim' name
303 END
304 writefile(import_no_as_lines, 'Ximport.vim')
305 assert_fails('source Ximport.vim', 'E488:', '', 2, 'Ximport.vim')
306
307 var import_invalid_string_lines =<< trim END
308 vim9script
309 import Xexport.vim
310 END
311 writefile(import_invalid_string_lines, 'Ximport.vim')
312 assert_fails('source Ximport.vim', 'E121:', '', 2, 'Ximport.vim')
313
314 var import_wrong_name_lines =<< trim END
315 vim9script
316 import './XnoExport.vim'
317 END
318 writefile(import_wrong_name_lines, 'Ximport.vim')
319 assert_fails('source Ximport.vim', 'E1053:', '', 2, 'Ximport.vim')
320
321 var import_redefining_lines =<< trim END
322 vim9script
323 import './Xexport.vim' as exported
324 var exported = 5
325 END
326 writefile(import_redefining_lines, 'Ximport.vim')
327 assert_fails('source Ximport.vim', 'E1213: Redefining imported item "exported"', '', 3)
328
Bram Moolenaar160aa862022-01-10 21:29:57 +0000329 var import_missing_dot_lines =<< trim END
330 vim9script
331 import './Xexport.vim' as expo
332 def Test()
333 expo = 9
334 enddef
335 defcompile
336 END
337 writefile(import_missing_dot_lines, 'Ximport.vim')
338 assert_fails('source Ximport.vim', 'E1258:', '', 1)
339
340 var import_missing_name_lines =<< trim END
341 vim9script
342 import './Xexport.vim' as expo
343 def Test()
344 expo.99 = 9
345 enddef
346 defcompile
347 END
348 writefile(import_missing_name_lines, 'Ximport.vim')
Bram Moolenaar76283822022-01-10 21:39:03 +0000349 assert_fails('source Ximport.vim', 'E1259:', '', 1)
Bram Moolenaar160aa862022-01-10 21:29:57 +0000350
Bram Moolenaard8448622022-01-07 21:39:52 +0000351 var import_assign_wrong_type_lines =<< trim END
352 vim9script
353 import './Xexport.vim' as expo
354 expo.exported = 'xxx'
355 END
356 writefile(import_assign_wrong_type_lines, 'Ximport.vim')
357 assert_fails('source Ximport.vim', 'E1012: Type mismatch; expected number but got string', '', 3)
358
359 var import_assign_const_lines =<< trim END
360 vim9script
361 import './Xexport.vim' as expo
362 expo.CONST = 4321
363 END
364 writefile(import_assign_const_lines, 'Ximport.vim')
365 assert_fails('source Ximport.vim', 'E46: Cannot change read-only variable "CONST"', '', 3)
366
367 delete('Ximport.vim')
368 delete('Ximport3.vim')
369 delete('Xexport.vim')
370
371 # Check that in a Vim9 script 'cpo' is set to the Vim default.
372 # Flags added or removed are also applied to the restored value.
373 set cpo=abcd
374 var lines =<< trim END
375 vim9script
376 g:cpo_in_vim9script = &cpo
377 set cpo+=f
378 set cpo-=c
379 g:cpo_after_vim9script = &cpo
380 END
381 writefile(lines, 'Xvim9_script')
382 source Xvim9_script
383 assert_equal('fabd', &cpo)
384 set cpo&vim
385 assert_equal(&cpo, g:cpo_in_vim9script)
386 var newcpo = substitute(&cpo, 'c', '', '') .. 'f'
387 assert_equal(newcpo, g:cpo_after_vim9script)
388
389 delete('Xvim9_script')
390enddef
391
392def Test_import_funcref()
393 var lines =<< trim END
394 vim9script
395 export def F(): number
396 return 42
397 enddef
398 export const G = F
399 END
400 writefile(lines, 'Xlib.vim')
401
402 lines =<< trim END
403 vim9script
404 import './Xlib.vim' as lib
405 const Foo = lib.G()
406 assert_equal(42, Foo)
407
408 def DoTest()
409 const Goo = lib.G()
410 assert_equal(42, Goo)
411 enddef
412 DoTest()
413 END
414 CheckScriptSuccess(lines)
415
416 delete('Xlib.vim')
417enddef
418
419def Test_import_fails()
420 writefile([], 'Xfoo.vim')
421 var lines =<< trim END
422 import './Xfoo.vim' as foo
423 foo = 'bar'
424 END
425 CheckDefAndScriptFailure(lines, ['E1094:', 'E1236: Cannot use foo itself'])
426 lines =<< trim END
427 vim9script
428 import './Xfoo.vim' as foo
429 var that = foo
430 END
431 CheckScriptFailure(lines, 'E1060: Expected dot after name: foo')
Bram Moolenaardd5893b2022-01-20 21:32:54 +0000432 lines =<< trim END
433 vim9script
434 import './Xfoo.vim' as foo
435 var that: any
436 that += foo
437 END
438 CheckScriptFailure(lines, 'E1060: Expected dot after name: foo')
439 lines =<< trim END
440 vim9script
441 import './Xfoo.vim' as foo
442 foo += 9
443 END
444 CheckScriptFailure(lines, 'E1060: Expected dot after name: foo')
Bram Moolenaard8448622022-01-07 21:39:52 +0000445
446 lines =<< trim END
447 vim9script
448 import './Xfoo.vim' as 9foo
449 END
450 CheckScriptFailure(lines, 'E1047:')
451 lines =<< trim END
452 vim9script
453 import './Xfoo.vim' as the#foo
454 END
455 CheckScriptFailure(lines, 'E1047:')
456 lines =<< trim END
457 vim9script
458 import './Xfoo.vim' as g:foo
459 END
460 CheckScriptFailure(lines, 'E1047:')
461
462 delete('Xfoo.vim')
463
464 lines =<< trim END
465 vim9script
466 def TheFunc()
467 echo 'the func'
468 enddef
469 export var Ref = TheFunc
470 END
471 writefile([], 'Xthat.vim')
Bram Moolenaar7c24dfd2022-01-08 17:03:55 +0000472
Bram Moolenaard8448622022-01-07 21:39:52 +0000473 lines =<< trim END
474 import './Xthat.vim' as That
475 That()
476 END
477 CheckDefAndScriptFailure(lines, ['E1094:', 'E1236: Cannot use That itself'])
Bram Moolenaar7c24dfd2022-01-08 17:03:55 +0000478
479 lines =<< trim END
Bram Moolenaar937610b2022-01-19 17:21:29 +0000480 vim9script
481 import './Xthat.vim' as That
482 def Func()
483 echo That()
484 enddef
485 Func()
486 END
487 CheckScriptFailure(lines, 'E1236: Cannot use That itself')
488
489 lines =<< trim END
Bram Moolenaar7c24dfd2022-01-08 17:03:55 +0000490 import './Xthat.vim' as one
491 import './Xthat.vim' as two
492 END
493 CheckScriptFailure(lines, 'E1262:')
494
495 delete('Xthat.vim')
Bram Moolenaard8448622022-01-07 21:39:52 +0000496
497 mkdir('Ximport')
498
499 writefile(['vim9script'], 'Ximport/.vim')
500 lines =<< trim END
501 vim9script
502 import './Ximport/.vim'
503 END
504 CheckScriptFailure(lines, 'E1261: Cannot import .vim without using "as"')
505 lines =<< trim END
506 vim9script
507 import './Ximport/.vim' as vim
508 END
509 CheckScriptSuccess(lines)
510
511 writefile(['vim9script'], 'Ximport/.vimrc')
512 lines =<< trim END
513 vim9script
514 import './Ximport/.vimrc'
515 END
516 CheckScriptFailure(lines, 'E1257: Imported script must use "as" or end in .vim')
517 lines =<< trim END
518 vim9script
519 import './Ximport/.vimrc' as vimrc
520 END
521 CheckScriptSuccess(lines)
522
523 delete('Ximport', 'rf')
524enddef
525
526func g:Trigger()
527 source Ximport.vim
528 return "echo 'yes'\<CR>"
529endfunc
530
531def Test_import_export_expr_map()
532 # check that :import and :export work when buffer is locked
533 var export_lines =<< trim END
534 vim9script
535 export def That(): string
536 return 'yes'
537 enddef
538 END
539 writefile(export_lines, 'Xexport_that.vim')
540
541 var import_lines =<< trim END
542 vim9script
543 import './Xexport_that.vim' as that
544 assert_equal('yes', that.That())
545 END
546 writefile(import_lines, 'Ximport.vim')
547
548 nnoremap <expr> trigger g:Trigger()
549 feedkeys('trigger', "xt")
550
551 delete('Xexport_that.vim')
552 delete('Ximport.vim')
553 nunmap trigger
554enddef
555
556def Test_import_in_filetype()
557 # check that :import works when the buffer is locked
558 mkdir('ftplugin', 'p')
559 var export_lines =<< trim END
560 vim9script
561 export var That = 'yes'
562 END
563 writefile(export_lines, 'ftplugin/Xexport_ft.vim')
564
565 var import_lines =<< trim END
566 vim9script
567 import './Xexport_ft.vim' as ft
568 assert_equal('yes', ft.That)
569 g:did_load_mytpe = 1
570 END
571 writefile(import_lines, 'ftplugin/qf.vim')
572
573 var save_rtp = &rtp
574 &rtp = getcwd() .. ',' .. &rtp
575
576 filetype plugin on
577 copen
578 assert_equal(1, g:did_load_mytpe)
579
580 quit!
581 delete('Xexport_ft.vim')
582 delete('ftplugin', 'rf')
583 &rtp = save_rtp
584enddef
585
586def Test_use_import_in_mapping()
587 var lines =<< trim END
588 vim9script
589 export def Funcx()
590 g:result = 42
591 enddef
592 END
593 writefile(lines, 'XsomeExport.vim')
594 lines =<< trim END
595 vim9script
596 import './XsomeExport.vim' as some
597 var Funcy = some.Funcx
598 nnoremap <F3> :call <sid>Funcy()<cr>
599 END
600 writefile(lines, 'Xmapscript.vim')
601
602 source Xmapscript.vim
603 feedkeys("\<F3>", "xt")
604 assert_equal(42, g:result)
605
606 unlet g:result
607 delete('XsomeExport.vim')
608 delete('Xmapscript.vim')
609 nunmap <F3>
610enddef
611
Bram Moolenaarf0e7e632022-01-21 13:29:56 +0000612def Test_use_import_in_command_completion()
Bram Moolenaar15d16352022-01-17 20:09:08 +0000613 var lines =<< trim END
614 vim9script
615 export def Complete(..._): list<string>
616 return ['abcd']
617 enddef
618 END
619 writefile(lines, 'Xscript.vim')
620
621 lines =<< trim END
622 vim9script
623 import './Xscript.vim'
624
625 command -nargs=1 -complete=customlist,Xscript.Complete Cmd echo 'ok'
626 feedkeys(":Cmd ab\<Tab>\<C-B>#\<CR>", 'xnt')
627 assert_equal('#Cmd abcd', @:)
628 END
629 CheckScriptSuccess(lines)
630
631 delcommand Cmd
632 delete('Xscript.vim')
633enddef
634
Bram Moolenaarf0e7e632022-01-21 13:29:56 +0000635def Test_use_autoload_import_in_insert_completion()
636 mkdir('Xdir/autoload', 'p')
637 var save_rtp = &rtp
638 exe 'set rtp^=' .. getcwd() .. '/Xdir'
639
640 var lines =<< trim END
641 vim9script
642 export def ThesaurusFunc(findbase: bool, _): any
643 if findbase
644 return 1
645 endif
646 return [
647 'check',
648 'experiment',
649 'test',
650 'verification'
651 ]
652 enddef
653 g:completion_loaded = 'yes'
654 END
655 writefile(lines, 'Xdir/autoload/completion.vim')
656
657 new
658 lines =<< trim END
659 vim9script
660 g:completion_loaded = 'no'
661 import autoload 'completion.vim'
662 set thesaurusfunc=completion.ThesaurusFunc
663 assert_equal('no', g:completion_loaded)
664 feedkeys("i\<C-X>\<C-T>\<C-N>\<Esc>", 'xt')
665 assert_equal('experiment', getline(1))
666 assert_equal('yes', g:completion_loaded)
667 END
668 CheckScriptSuccess(lines)
669
670 set thesaurusfunc=
671 bwipe!
672 delete('Xdir', 'rf')
673 &rtp = save_rtp
674enddef
675
Bram Moolenaare70dd112022-01-21 16:31:11 +0000676def Test_use_autoload_import_in_fold_expression()
677 mkdir('Xdir/autoload', 'p')
678 var save_rtp = &rtp
679 exe 'set rtp^=' .. getcwd() .. '/Xdir'
680
681 var lines =<< trim END
682 vim9script
683 export def Expr(): string
684 return getline(v:lnum) =~ '^#' ? '>1' : '1'
685 enddef
Bram Moolenaar9530b582022-01-22 13:39:08 +0000686 export def Text(): string
687 return 'fold text'
688 enddef
Bram Moolenaare70dd112022-01-21 16:31:11 +0000689 g:fold_loaded = 'yes'
690 END
691 writefile(lines, 'Xdir/autoload/fold.vim')
692
693 lines =<< trim END
694 vim9script
695 import autoload 'fold.vim'
696 &foldexpr = 'fold.Expr()'
Bram Moolenaar9530b582022-01-22 13:39:08 +0000697 &foldtext = 'fold.Text()'
Bram Moolenaare70dd112022-01-21 16:31:11 +0000698 &foldmethod = 'expr'
699 &debug = 'throw'
700 END
701 new
702 setline(1, ['# one', 'text', '# two', 'text'])
703 g:fold_loaded = 'no'
704 CheckScriptSuccess(lines)
705 assert_equal('no', g:fold_loaded)
706 redraw
707 assert_equal('yes', g:fold_loaded)
708
709 # Check that script context of 'foldexpr' is copied to another buffer.
710 edit! otherfile
711 redraw
712
Bram Moolenaar9530b582022-01-22 13:39:08 +0000713 set foldexpr= foldtext& foldmethod& debug=
Bram Moolenaare70dd112022-01-21 16:31:11 +0000714 bwipe!
715 delete('Xdir', 'rf')
716 &rtp = save_rtp
717enddef
718
Bram Moolenaard8448622022-01-07 21:39:52 +0000719def Test_export_fails()
720 CheckScriptFailure(['export var some = 123'], 'E1042:')
721 CheckScriptFailure(['vim9script', 'export var g:some'], 'E1022:')
722 CheckScriptFailure(['vim9script', 'export echo 134'], 'E1043:')
723
724 assert_fails('export something', 'E1043:')
725enddef
726
727func Test_import_fails_without_script()
728 CheckRunVimInTerminal
729
730 " call indirectly to avoid compilation error for missing functions
731 call Run_Test_import_fails_on_command_line()
732endfunc
733
734def Run_Test_import_fails_on_command_line()
735 var export =<< trim END
736 vim9script
737 export def Foo(): number
738 return 0
739 enddef
740 END
741 writefile(export, 'XexportCmd.vim')
742
743 var buf = RunVimInTerminal('-c "import Foo from ''./XexportCmd.vim''"', {
744 rows: 6, wait_for_ruler: 0})
745 WaitForAssert(() => assert_match('^E1094:', term_getline(buf, 5)))
746
747 delete('XexportCmd.vim')
748 StopVimInTerminal(buf)
749enddef
750
751def Test_vim9_reload_noclear()
752 var lines =<< trim END
753 vim9script
754 export var exported = 'thexport'
755
756 export def TheFunc(x = 0)
757 enddef
758 END
759 writefile(lines, 'XExportReload')
760 lines =<< trim END
761 vim9script noclear
762 g:loadCount += 1
763 var s:reloaded = 'init'
764 import './XExportReload' as exp
765
766 def Again(): string
767 return 'again'
768 enddef
769
770 exp.TheFunc()
771
772 if exists('s:loaded') | finish | endif
773 var s:loaded = true
774
775 var s:notReloaded = 'yes'
776 s:reloaded = 'first'
777 def g:Values(): list<string>
778 return [s:reloaded, s:notReloaded, Again(), Once(), exp.exported]
779 enddef
780
781 def Once(): string
782 return 'once'
783 enddef
784 END
785 writefile(lines, 'XReloaded')
786 g:loadCount = 0
787 source XReloaded
788 assert_equal(1, g:loadCount)
789 assert_equal(['first', 'yes', 'again', 'once', 'thexport'], g:Values())
790 source XReloaded
791 assert_equal(2, g:loadCount)
792 assert_equal(['init', 'yes', 'again', 'once', 'thexport'], g:Values())
793 source XReloaded
794 assert_equal(3, g:loadCount)
795 assert_equal(['init', 'yes', 'again', 'once', 'thexport'], g:Values())
796
797 delete('XReloaded')
798 delete('XExportReload')
799 delfunc g:Values
800 unlet g:loadCount
801
802 lines =<< trim END
803 vim9script
804 def Inner()
805 enddef
806 END
807 lines->writefile('XreloadScript.vim')
808 source XreloadScript.vim
809
810 lines =<< trim END
811 vim9script
812 def Outer()
813 def Inner()
814 enddef
815 enddef
816 defcompile
817 END
818 lines->writefile('XreloadScript.vim')
819 source XreloadScript.vim
820
821 delete('XreloadScript.vim')
822enddef
823
824def Test_vim9_reload_import()
825 var lines =<< trim END
826 vim9script
827 const var = ''
828 var valone = 1234
829 def MyFunc(arg: string)
830 valone = 5678
831 enddef
832 END
833 var morelines =<< trim END
834 var valtwo = 222
835 export def GetValtwo(): number
836 return valtwo
837 enddef
838 END
839 writefile(lines + morelines, 'Xreload.vim')
840 source Xreload.vim
841 source Xreload.vim
842 source Xreload.vim
843
844 # cannot declare a var twice
845 lines =<< trim END
846 vim9script
847 var valone = 1234
848 var valone = 5678
849 END
850 writefile(lines, 'Xreload.vim')
851 assert_fails('source Xreload.vim', 'E1041:', '', 3, 'Xreload.vim')
852
853 delete('Xreload.vim')
854 delete('Ximport.vim')
855enddef
856
857" if a script is reloaded with a script-local variable that changed its type, a
858" compiled function using that variable must fail.
859def Test_script_reload_change_type()
860 var lines =<< trim END
861 vim9script noclear
862 var str = 'string'
863 def g:GetStr(): string
864 return str .. 'xxx'
865 enddef
866 END
867 writefile(lines, 'Xreload.vim')
868 source Xreload.vim
869 echo g:GetStr()
870
871 lines =<< trim END
872 vim9script noclear
873 var str = 1234
874 END
875 writefile(lines, 'Xreload.vim')
876 source Xreload.vim
877 assert_fails('echo g:GetStr()', 'E1150:')
878
879 delfunc g:GetStr
880 delete('Xreload.vim')
881enddef
882
883" Define CallFunc so that the test can be compiled
884command CallFunc echo 'nop'
885
886def Test_script_reload_from_function()
887 var lines =<< trim END
888 vim9script
889
890 if exists('g:loaded')
891 finish
892 endif
893 g:loaded = 1
894 delcommand CallFunc
895 command CallFunc Func()
896 def Func()
897 so XreloadFunc.vim
898 g:didTheFunc = 1
899 enddef
900 END
901 writefile(lines, 'XreloadFunc.vim')
902 source XreloadFunc.vim
903 CallFunc
904 assert_equal(1, g:didTheFunc)
905
906 delete('XreloadFunc.vim')
907 delcommand CallFunc
908 unlet g:loaded
909 unlet g:didTheFunc
910enddef
911
912def s:RetSome(): string
913 return 'some'
914enddef
915
916" Not exported function that is referenced needs to be accessed by the
917" script-local name.
918def Test_vim9_funcref()
919 var sortlines =<< trim END
920 vim9script
921 def Compare(i1: number, i2: number): number
922 return i2 - i1
923 enddef
924
925 export def FastSort(): list<number>
926 return range(5)->sort(Compare)
927 enddef
928
929 export def GetString(arg: string): string
930 return arg
931 enddef
932 END
933 writefile(sortlines, 'Xsort.vim')
934
935 var lines =<< trim END
936 vim9script
937 import './Xsort.vim'
938 def Test()
939 g:result = Xsort.FastSort()
940 enddef
941 Test()
Bram Moolenaar7c24dfd2022-01-08 17:03:55 +0000942 END
943 writefile(lines, 'Xscript.vim')
944 source Xscript.vim
945 assert_equal([4, 3, 2, 1, 0], g:result)
946 unlet g:result
Bram Moolenaard8448622022-01-07 21:39:52 +0000947
Bram Moolenaar7c24dfd2022-01-08 17:03:55 +0000948 lines =<< trim END
949 vim9script
Bram Moolenaard8448622022-01-07 21:39:52 +0000950 # using a function imported with "as"
951 import './Xsort.vim' as anAlias
952 assert_equal('yes', anAlias.GetString('yes'))
953
954 # using the function from a compiled function
955 def TestMore(): string
956 var s = s:anAlias.GetString('foo')
957 return s .. anAlias.GetString('bar')
958 enddef
959 assert_equal('foobar', TestMore())
960
961 # error when using a function that isn't exported
962 assert_fails('anAlias.Compare(1, 2)', 'E1049:')
963 END
964 writefile(lines, 'Xscript.vim')
965
Bram Moolenaard8448622022-01-07 21:39:52 +0000966 delete('Xsort.vim')
967 delete('Xscript.vim')
968
969 var Funcref = function('s:RetSome')
970 assert_equal('some', Funcref())
971enddef
972
973" Check that when searching for "FilterFunc" it finds the import in the
974" script where FastFilter() is called from, both as a string and as a direct
975" function reference.
976def Test_vim9_funcref_other_script()
977 var filterLines =<< trim END
978 vim9script
979 export def FilterFunc(idx: number, val: number): bool
980 return idx % 2 == 1
981 enddef
982 export def FastFilter(): list<number>
983 return range(10)->filter('FilterFunc(v:key, v:val)')
984 enddef
985 export def FastFilterDirect(): list<number>
986 return range(10)->filter(FilterFunc)
987 enddef
988 END
989 writefile(filterLines, 'Xfilter.vim')
990
991 var lines =<< trim END
992 vim9script
993 import './Xfilter.vim' as filter
994 def Test()
995 var x: list<number> = filter.FastFilter()
996 enddef
997 Test()
998 def TestDirect()
999 var x: list<number> = filter.FastFilterDirect()
1000 enddef
1001 TestDirect()
1002 END
1003 CheckScriptSuccess(lines)
1004 delete('Xfilter.vim')
1005enddef
1006
1007def Test_import_absolute()
1008 var import_lines = [
1009 'vim9script',
1010 'import "' .. escape(getcwd(), '\') .. '/Xexport_abs.vim" as abs',
1011 'def UseExported()',
1012 ' g:imported_abs = abs.exported',
1013 ' abs.exported = 8888',
1014 ' g:imported_after = abs.exported',
1015 'enddef',
1016 'UseExported()',
1017 'g:import_disassembled = execute("disass UseExported")',
1018 ]
1019 writefile(import_lines, 'Ximport_abs.vim')
1020 writefile(s:export_script_lines, 'Xexport_abs.vim')
1021
1022 source Ximport_abs.vim
1023
1024 assert_equal(9876, g:imported_abs)
1025 assert_equal(8888, g:imported_after)
1026 assert_match('<SNR>\d\+_UseExported\_s*' ..
1027 'g:imported_abs = abs.exported\_s*' ..
1028 '0 LOADSCRIPT exported-2 from .*Xexport_abs.vim\_s*' ..
1029 '1 STOREG g:imported_abs\_s*' ..
1030 'abs.exported = 8888\_s*' ..
1031 '2 PUSHNR 8888\_s*' ..
1032 '3 STORESCRIPT exported-2 in .*Xexport_abs.vim\_s*' ..
1033 'g:imported_after = abs.exported\_s*' ..
1034 '4 LOADSCRIPT exported-2 from .*Xexport_abs.vim\_s*' ..
1035 '5 STOREG g:imported_after',
1036 g:import_disassembled)
1037
1038 Undo_export_script_lines()
1039 unlet g:imported_abs
1040 unlet g:import_disassembled
1041
1042 delete('Ximport_abs.vim')
1043 delete('Xexport_abs.vim')
1044enddef
1045
1046def Test_import_rtp()
1047 var import_lines = [
1048 'vim9script',
1049 'import "Xexport_rtp.vim" as rtp',
1050 'g:imported_rtp = rtp.exported',
1051 ]
1052 writefile(import_lines, 'Ximport_rtp.vim')
1053 mkdir('import', 'p')
1054 writefile(s:export_script_lines, 'import/Xexport_rtp.vim')
1055
1056 var save_rtp = &rtp
1057 &rtp = getcwd()
1058 source Ximport_rtp.vim
1059 &rtp = save_rtp
1060
1061 assert_equal(9876, g:imported_rtp)
1062
1063 Undo_export_script_lines()
1064 unlet g:imported_rtp
1065 delete('Ximport_rtp.vim')
1066 delete('import', 'rf')
1067enddef
1068
1069def Test_import_compile_error()
1070 var export_lines = [
1071 'vim9script',
1072 'export def ExpFunc(): string',
1073 ' return notDefined',
1074 'enddef',
1075 ]
1076 writefile(export_lines, 'Xexported.vim')
1077
1078 var import_lines = [
1079 'vim9script',
1080 'import "./Xexported.vim" as expo',
1081 'def ImpFunc()',
1082 ' echo expo.ExpFunc()',
1083 'enddef',
1084 'defcompile',
1085 ]
1086 writefile(import_lines, 'Ximport.vim')
1087
1088 try
1089 source Ximport.vim
1090 catch /E1001/
1091 # Error should be before the Xexported.vim file.
1092 assert_match('E1001: Variable not found: notDefined', v:exception)
1093 assert_match('function <SNR>\d\+_ImpFunc\[1\]..<SNR>\d\+_ExpFunc, line 1', v:throwpoint)
1094 endtry
1095
1096 delete('Xexported.vim')
1097 delete('Ximport.vim')
1098enddef
1099
1100def Test_func_overrules_import_fails()
1101 var export_lines =<< trim END
1102 vim9script
1103 export def Func()
1104 echo 'imported'
1105 enddef
1106 END
1107 writefile(export_lines, 'XexportedFunc.vim')
1108
1109 var lines =<< trim END
1110 vim9script
1111 import './XexportedFunc.vim' as Func
1112 def Func()
1113 echo 'local to function'
1114 enddef
1115 END
Bram Moolenaar937610b2022-01-19 17:21:29 +00001116 CheckScriptFailure(lines, 'E1213: Redefining imported item "Func"')
Bram Moolenaard8448622022-01-07 21:39:52 +00001117
1118 lines =<< trim END
1119 vim9script
1120 import './XexportedFunc.vim' as Func
1121 def Outer()
1122 def Func()
1123 echo 'local to function'
1124 enddef
1125 enddef
1126 defcompile
1127 END
1128 CheckScriptFailure(lines, 'E1236:')
1129
1130 delete('XexportedFunc.vim')
1131enddef
1132
1133def Test_source_vim9_from_legacy()
1134 var vim9_lines =<< trim END
1135 vim9script
1136 var local = 'local'
1137 g:global = 'global'
1138 export var exported = 'exported'
1139 export def GetText(): string
1140 return 'text'
1141 enddef
1142 END
1143 writefile(vim9_lines, 'Xvim9_script.vim')
1144
1145 var legacy_lines =<< trim END
1146 source Xvim9_script.vim
1147
1148 call assert_false(exists('local'))
1149 call assert_false(exists('exported'))
1150 call assert_false(exists('s:exported'))
1151 call assert_equal('global', global)
1152 call assert_equal('global', g:global)
Bram Moolenaard8448622022-01-07 21:39:52 +00001153 END
1154 writefile(legacy_lines, 'Xlegacy_script.vim')
1155
1156 source Xlegacy_script.vim
1157 assert_equal('global', g:global)
1158 unlet g:global
1159
1160 delete('Xlegacy_script.vim')
1161 delete('Xvim9_script.vim')
1162enddef
1163
Bram Moolenaarc43e6232022-01-13 20:51:56 +00001164def Test_import_vim9_from_legacy()
1165 var vim9_lines =<< trim END
1166 vim9script
1167 var local = 'local'
1168 g:global = 'global'
1169 export var exported = 'exported'
1170 export def GetText(): string
1171 return 'text'
1172 enddef
1173 END
1174 writefile(vim9_lines, 'Xvim9_export.vim')
1175
1176 var legacy_lines =<< trim END
1177 import './Xvim9_export.vim' as vim9
1178
1179 call assert_false(exists('vim9'))
1180 call assert_false(exists('local'))
1181 call assert_false(exists('s:vim9.local'))
1182 call assert_equal('global', global)
1183 call assert_equal('global', g:global)
1184 call assert_false(exists('exported'))
1185 call assert_false(exists('s:exported'))
1186 call assert_false(exists('*GetText'))
1187
1188 " imported symbol is script-local
1189 call assert_equal('exported', s:vim9.exported)
1190 call assert_equal('text', s:vim9.GetText())
1191 END
1192 writefile(legacy_lines, 'Xlegacy_script.vim')
1193
1194 source Xlegacy_script.vim
1195 assert_equal('global', g:global)
1196 unlet g:global
1197
1198 delete('Xlegacy_script.vim')
1199 delete('Xvim9_export.vim')
1200enddef
1201
Bram Moolenaard8448622022-01-07 21:39:52 +00001202def Test_cmdline_win()
1203 # if the Vim syntax highlighting uses Vim9 constructs they can be used from
1204 # the command line window.
1205 mkdir('rtp/syntax', 'p')
1206 var export_lines =<< trim END
1207 vim9script
1208 export var That = 'yes'
1209 END
1210 writefile(export_lines, 'rtp/syntax/Xexport.vim')
1211 var import_lines =<< trim END
1212 vim9script
1213 import './Xexport.vim' as exp
1214 echo exp.That
1215 END
1216 writefile(import_lines, 'rtp/syntax/vim.vim')
1217 var save_rtp = &rtp
1218 &rtp = getcwd() .. '/rtp' .. ',' .. &rtp
1219 syntax on
1220 augroup CmdWin
1221 autocmd CmdwinEnter * g:got_there = 'yes'
1222 augroup END
1223 # this will open and also close the cmdline window
1224 feedkeys('q:', 'xt')
1225 assert_equal('yes', g:got_there)
1226
1227 augroup CmdWin
1228 au!
1229 augroup END
1230 &rtp = save_rtp
1231 delete('rtp', 'rf')
1232enddef
1233
1234def Test_import_gone_when_sourced_twice()
1235 var exportlines =<< trim END
1236 vim9script
1237 if exists('g:guard')
1238 finish
1239 endif
1240 g:guard = 1
1241 export var name = 'someName'
1242 END
1243 writefile(exportlines, 'XexportScript.vim')
1244
1245 var lines =<< trim END
1246 vim9script
1247 import './XexportScript.vim' as expo
1248 def g:GetName(): string
1249 return expo.name
1250 enddef
1251 END
1252 writefile(lines, 'XscriptImport.vim')
1253 so XscriptImport.vim
1254 assert_equal('someName', g:GetName())
1255
1256 so XexportScript.vim
1257 assert_fails('call g:GetName()', 'E1149:')
1258
1259 delfunc g:GetName
1260 delete('XexportScript.vim')
1261 delete('XscriptImport.vim')
1262 unlet g:guard
1263enddef
1264
Bram Moolenaar160aa862022-01-10 21:29:57 +00001265" test using an auto-loaded function and variable
Bram Moolenaar0e3e7ba2022-01-13 20:18:56 +00001266def Test_vim9_autoload_full_name()
Bram Moolenaar160aa862022-01-10 21:29:57 +00001267 var lines =<< trim END
1268 vim9script
1269 def some#gettest(): string
1270 return 'test'
1271 enddef
1272 g:some#name = 'name'
1273 g:some#dict = {key: 'value'}
1274
1275 def some#varargs(a1: string, ...l: list<string>): string
1276 return a1 .. l[0] .. l[1]
1277 enddef
1278 END
1279
1280 mkdir('Xdir/autoload', 'p')
1281 writefile(lines, 'Xdir/autoload/some.vim')
1282 var save_rtp = &rtp
1283 exe 'set rtp^=' .. getcwd() .. '/Xdir'
1284
1285 assert_equal('test', g:some#gettest())
1286 assert_equal('name', g:some#name)
1287 assert_equal('value', g:some#dict.key)
1288 g:some#other = 'other'
1289 assert_equal('other', g:some#other)
1290
1291 assert_equal('abc', some#varargs('a', 'b', 'c'))
1292
1293 # upper case script name works
1294 lines =<< trim END
1295 vim9script
1296 def Other#getOther(): string
1297 return 'other'
1298 enddef
1299 END
1300 writefile(lines, 'Xdir/autoload/Other.vim')
1301 assert_equal('other', g:Other#getOther())
1302
1303 delete('Xdir', 'rf')
1304 &rtp = save_rtp
1305enddef
1306
1307def Test_vim9script_autoload()
1308 mkdir('Xdir/autoload', 'p')
1309 var save_rtp = &rtp
1310 exe 'set rtp^=' .. getcwd() .. '/Xdir'
1311
Bram Moolenaarfd218c82022-01-18 16:26:24 +00001312 # when the path has "/autoload/" prefix is not needed
Bram Moolenaar160aa862022-01-10 21:29:57 +00001313 var lines =<< trim END
Bram Moolenaarfd218c82022-01-18 16:26:24 +00001314 vim9script
Bram Moolenaar17d36cb2022-01-12 11:46:40 +00001315 g:prefixed_loaded += 1
Bram Moolenaar160aa862022-01-10 21:29:57 +00001316
1317 export def Gettest(): string
1318 return 'test'
1319 enddef
1320
Bram Moolenaar0e3e7ba2022-01-13 20:18:56 +00001321 export var name = 'name'
1322
1323 export func GetFunc()
1324 return Gettest() .. 'more' .. s:name
Bram Moolenaar160aa862022-01-10 21:29:57 +00001325 endfunc
1326
Bram Moolenaar0e3e7ba2022-01-13 20:18:56 +00001327 export def GetDef(): string
1328 return Gettest() .. 'more' .. name
1329 enddef
1330
Bram Moolenaar160aa862022-01-10 21:29:57 +00001331 export final fname = 'final'
1332 export const cname = 'const'
1333 END
1334 writefile(lines, 'Xdir/autoload/prefixed.vim')
1335
Bram Moolenaar17d36cb2022-01-12 11:46:40 +00001336 g:prefixed_loaded = 0
1337 g:expected_loaded = 0
Bram Moolenaar160aa862022-01-10 21:29:57 +00001338 lines =<< trim END
1339 vim9script
1340 import autoload 'prefixed.vim'
Bram Moolenaar17d36cb2022-01-12 11:46:40 +00001341 assert_equal(g:expected_loaded, g:prefixed_loaded)
Bram Moolenaar160aa862022-01-10 21:29:57 +00001342 assert_equal('test', prefixed.Gettest())
Bram Moolenaar17d36cb2022-01-12 11:46:40 +00001343 assert_equal(1, g:prefixed_loaded)
Bram Moolenaar160aa862022-01-10 21:29:57 +00001344
Bram Moolenaar0e3e7ba2022-01-13 20:18:56 +00001345 assert_equal('testmorename', prefixed.GetFunc())
1346 assert_equal('testmorename', prefixed.GetDef())
Bram Moolenaar160aa862022-01-10 21:29:57 +00001347 assert_equal('name', prefixed.name)
1348 assert_equal('final', prefixed.fname)
1349 assert_equal('const', prefixed.cname)
1350 END
1351 CheckScriptSuccess(lines)
Bram Moolenaar17d36cb2022-01-12 11:46:40 +00001352 # can source it again, autoload script not loaded again
1353 g:expected_loaded = 1
1354 CheckScriptSuccess(lines)
Bram Moolenaar160aa862022-01-10 21:29:57 +00001355
1356 # can also get the items by autoload name
1357 lines =<< trim END
1358 call assert_equal('test', prefixed#Gettest())
Bram Moolenaar0e3e7ba2022-01-13 20:18:56 +00001359 call assert_equal('testmorename', prefixed#GetFunc())
Bram Moolenaar160aa862022-01-10 21:29:57 +00001360 call assert_equal('name', prefixed#name)
1361 call assert_equal('final', prefixed#fname)
1362 call assert_equal('const', prefixed#cname)
1363 END
1364 CheckScriptSuccess(lines)
1365
Bram Moolenaarf111cdf2022-01-12 12:48:17 +00001366 unlet g:prefixed_loaded
1367 unlet g:expected_loaded
1368 delete('Xdir', 'rf')
1369 &rtp = save_rtp
1370enddef
1371
Bram Moolenaard02dce22022-01-18 17:43:04 +00001372def Test_import_autoload_not_exported()
1373 mkdir('Xdir/autoload', 'p')
1374 var save_rtp = &rtp
1375 exe 'set rtp^=' .. getcwd() .. '/Xdir'
1376
1377 # error when using an item that is not exported from an autoload script
1378 var exportLines =<< trim END
1379 vim9script
1380 var notExported = 123
1381 def NotExport()
1382 echo 'nop'
1383 enddef
1384 END
1385 writefile(exportLines, 'Xdir/autoload/notExport1.vim')
1386
1387 var lines =<< trim END
1388 vim9script
1389 import autoload 'notExport1.vim'
1390 echo notExport1.notFound
1391 END
1392 CheckScriptFailure(lines, 'E1048: Item not found in script: notFound')
1393
1394 lines =<< trim END
1395 vim9script
1396 import autoload 'notExport1.vim'
1397 echo notExport1.notExported
1398 END
1399 CheckScriptFailure(lines, 'E1049: Item not exported in script: notExported')
1400
1401 lines =<< trim END
1402 vim9script
1403 import autoload 'notExport1.vim'
1404 echo notExport1.NotFunc()
1405 END
1406 CheckScriptFailure(lines, 'E1048: Item not found in script: NotFunc')
1407
1408 lines =<< trim END
1409 vim9script
1410 import autoload 'notExport1.vim'
1411 echo notExport1.NotExport()
1412 END
1413 CheckScriptFailure(lines, 'E1049: Item not exported in script: NotExport')
1414
1415 lines =<< trim END
1416 vim9script
1417 import autoload 'notExport1.vim'
1418 echo 'text'->notExport1.NotFunc()
1419 END
1420 CheckScriptFailure(lines, 'E1048: Item not found in script: NotFunc')
1421
1422 lines =<< trim END
1423 vim9script
1424 import autoload 'notExport1.vim'
1425 echo 'text'->notExport1.NotExport()
1426 END
1427 CheckScriptFailure(lines, 'E1049: Item not exported in script: NotExport')
1428
1429 # using a :def function we use a different autoload script every time so that
1430 # the function is compiled without the script loaded
1431 writefile(exportLines, 'Xdir/autoload/notExport2.vim')
1432 lines =<< trim END
1433 vim9script
1434 import autoload 'notExport2.vim'
1435 def Testit()
1436 echo notExport2.notFound
1437 enddef
1438 Testit()
1439 END
1440 CheckScriptFailure(lines, 'E1048: Item not found in script: notExport2#notFound')
1441
1442 writefile(exportLines, 'Xdir/autoload/notExport3.vim')
1443 lines =<< trim END
1444 vim9script
1445 import autoload 'notExport3.vim'
1446 def Testit()
1447 echo notExport3.notExported
1448 enddef
1449 Testit()
1450 END
1451 # don't get E1049 because it is too complicated to figure out
1452 CheckScriptFailure(lines, 'E1048: Item not found in script: notExport3#notExported')
1453
1454 writefile(exportLines, 'Xdir/autoload/notExport4.vim')
1455 lines =<< trim END
1456 vim9script
1457 import autoload 'notExport4.vim'
1458 def Testit()
1459 echo notExport4.NotFunc()
1460 enddef
1461 Testit()
1462 END
1463 CheckScriptFailure(lines, 'E117: Unknown function: notExport4#NotFunc')
1464
1465 writefile(exportLines, 'Xdir/autoload/notExport5.vim')
1466 lines =<< trim END
1467 vim9script
1468 import autoload 'notExport5.vim'
1469 def Testit()
1470 echo notExport5.NotExport()
1471 enddef
1472 Testit()
1473 END
1474 CheckScriptFailure(lines, 'E117: Unknown function: notExport5#NotExport')
1475
1476 writefile(exportLines, 'Xdir/autoload/notExport6.vim')
1477 lines =<< trim END
1478 vim9script
1479 import autoload 'notExport6.vim'
1480 def Testit()
1481 echo 'text'->notExport6.NotFunc()
1482 enddef
1483 Testit()
1484 END
1485 CheckScriptFailure(lines, 'E117: Unknown function: notExport6#NotFunc')
1486
1487 writefile(exportLines, 'Xdir/autoload/notExport7.vim')
1488 lines =<< trim END
1489 vim9script
1490 import autoload 'notExport7.vim'
1491 def Testit()
1492 echo 'text'->notExport7.NotExport()
1493 enddef
1494 Testit()
1495 END
1496 CheckScriptFailure(lines, 'E117: Unknown function: notExport7#NotExport')
1497
1498 delete('Xdir', 'rf')
1499 &rtp = save_rtp
1500enddef
1501
Bram Moolenaarf111cdf2022-01-12 12:48:17 +00001502def Test_vim9script_autoload_call()
1503 mkdir('Xdir/autoload', 'p')
1504 var save_rtp = &rtp
1505 exe 'set rtp^=' .. getcwd() .. '/Xdir'
1506
1507 var lines =<< trim END
Bram Moolenaarfd218c82022-01-18 16:26:24 +00001508 vim9script
Bram Moolenaarf111cdf2022-01-12 12:48:17 +00001509
Bram Moolenaarcbbc48f2022-01-18 12:58:28 +00001510 export def RetArg(arg: string): string
1511 return arg
1512 enddef
1513
Bram Moolenaarf111cdf2022-01-12 12:48:17 +00001514 export def Getother()
1515 g:result = 'other'
1516 enddef
1517 END
Bram Moolenaar5d982692022-01-12 15:15:27 +00001518 writefile(lines, 'Xdir/autoload/another.vim')
Bram Moolenaarf111cdf2022-01-12 12:48:17 +00001519
1520 lines =<< trim END
1521 vim9script
Bram Moolenaar5d982692022-01-12 15:15:27 +00001522 import autoload 'another.vim'
Bram Moolenaarcbbc48f2022-01-18 12:58:28 +00001523
1524 # compile this before 'another.vim' is loaded
1525 def CallAnother()
1526 assert_equal('foo', 'foo'->another.RetArg())
1527 enddef
1528 CallAnother()
1529
Bram Moolenaar5d982692022-01-12 15:15:27 +00001530 call another.Getother()
Bram Moolenaarf111cdf2022-01-12 12:48:17 +00001531 assert_equal('other', g:result)
Bram Moolenaar3d8e25a2022-01-22 11:00:02 +00001532
1533 assert_equal('arg', call('another.RetArg', ['arg']))
Bram Moolenaarf111cdf2022-01-12 12:48:17 +00001534 END
1535 CheckScriptSuccess(lines)
1536
1537 unlet g:result
Bram Moolenaar160aa862022-01-10 21:29:57 +00001538 delete('Xdir', 'rf')
1539 &rtp = save_rtp
1540enddef
1541
Bram Moolenaarb697dc22022-01-22 11:27:29 +00001542def Test_vim9script_noclear_autoload()
1543 mkdir('Xdir/autoload', 'p')
1544 var save_rtp = &rtp
1545 exe 'set rtp^=' .. getcwd() .. '/Xdir'
1546
1547 var lines =<< trim END
1548 vim9script
1549 export def Func(): string
1550 return 'called'
1551 enddef
1552 g:double_loaded = 'yes'
1553 END
1554 writefile(lines, 'Xdir/autoload/double.vim')
1555
1556 lines =<< trim END
1557 vim9script noclear
1558 if exists('g:script_loaded')
1559 finish
1560 endif
1561 g:script_loaded = true
1562
1563 import autoload 'double.vim'
1564 nnoremap <F3> <ScriptCmd>g:result = double.Func()<CR>
1565 END
1566 g:double_loaded = 'no'
1567 writefile(lines, 'Xloaddouble')
1568 source Xloaddouble
1569 assert_equal('no', g:double_loaded)
1570 assert_equal(true, g:script_loaded)
1571 source Xloaddouble
1572 feedkeys("\<F3>", 'xt')
1573 assert_equal('called', g:result)
1574 assert_equal('yes', g:double_loaded)
1575
1576 delete('Xloaddouble')
1577 unlet g:double_loaded
1578 unlet g:script_loaded
1579 unlet g:result
1580 delete('Xdir', 'rf')
1581 &rtp = save_rtp
1582enddef
1583
Bram Moolenaar9c7cae62022-01-20 19:10:25 +00001584def Test_vim9script_autoload_duplicate()
1585 mkdir('Xdir/autoload', 'p')
1586
1587 var lines =<< trim END
1588 vim9script
1589
1590 export def Func()
1591 enddef
1592
1593 def Func()
1594 enddef
1595 END
1596 writefile(lines, 'Xdir/autoload/dupfunc.vim')
1597 assert_fails('source Xdir/autoload/dupfunc.vim', 'E1073:')
1598
1599 lines =<< trim END
1600 vim9script
1601
1602 def Func()
1603 enddef
1604
1605 export def Func()
1606 enddef
1607 END
1608 writefile(lines, 'Xdir/autoload/dup2func.vim')
1609 assert_fails('source Xdir/autoload/dup2func.vim', 'E1073:')
1610
1611 lines =<< trim END
1612 vim9script
1613
1614 def Func()
1615 enddef
1616
1617 export var Func = 'asdf'
1618 END
1619 writefile(lines, 'Xdir/autoload/dup3func.vim')
1620 assert_fails('source Xdir/autoload/dup3func.vim', 'E1041: Redefining script item Func')
1621
1622 lines =<< trim END
1623 vim9script
1624
1625 export var Func = 'asdf'
1626
1627 def Func()
1628 enddef
1629 END
1630 writefile(lines, 'Xdir/autoload/dup4func.vim')
1631 assert_fails('source Xdir/autoload/dup4func.vim', 'E707:')
1632
1633 lines =<< trim END
1634 vim9script
1635
1636 var Func = 'asdf'
1637
1638 export def Func()
1639 enddef
1640 END
1641 writefile(lines, 'Xdir/autoload/dup5func.vim')
1642 assert_fails('source Xdir/autoload/dup5func.vim', 'E707:')
1643
1644 lines =<< trim END
1645 vim9script
1646
1647 export def Func()
1648 enddef
1649
1650 var Func = 'asdf'
1651 END
1652 writefile(lines, 'Xdir/autoload/dup6func.vim')
1653 assert_fails('source Xdir/autoload/dup6func.vim', 'E1041: Redefining script item Func')
1654
1655 delete('Xdir', 'rf')
1656enddef
1657
Bram Moolenaar2017d6f2022-01-20 19:38:46 +00001658def Test_autoload_missing_function_name()
1659 mkdir('Xdir/autoload', 'p')
1660
1661 var lines =<< trim END
1662 vim9script
1663
1664 def loadme#()
1665 enddef
1666 END
1667 writefile(lines, 'Xdir/autoload/loadme.vim')
1668 assert_fails('source Xdir/autoload/loadme.vim', 'E129:')
1669
1670 delete('Xdir', 'rf')
1671enddef
1672
Bram Moolenaar19e69a62022-01-21 20:37:05 +00001673def Test_autoload_name_wring()
1674 var lines =<< trim END
1675 vim9script
1676 def Xscriptname#Func()
1677 enddef
1678 END
1679 writefile(lines, 'Xscriptname.vim')
1680 CheckScriptFailure(lines, 'E1263:')
1681
1682 delete('Xscriptname')
1683enddef
1684
Bram Moolenaard041f422022-01-12 19:54:00 +00001685def Test_import_autoload_postponed()
1686 mkdir('Xdir/autoload', 'p')
1687 var save_rtp = &rtp
1688 exe 'set rtp^=' .. getcwd() .. '/Xdir'
1689
1690 var lines =<< trim END
Bram Moolenaarfd218c82022-01-18 16:26:24 +00001691 vim9script
Bram Moolenaard041f422022-01-12 19:54:00 +00001692
1693 g:loaded_postponed = 'true'
1694 export var variable = 'bla'
1695 export def Function(): string
1696 return 'bla'
1697 enddef
1698 END
1699 writefile(lines, 'Xdir/autoload/postponed.vim')
1700
1701 lines =<< trim END
1702 vim9script
1703
1704 import autoload 'postponed.vim'
1705 def Tryit()
1706 echo postponed.variable
1707 echo postponed.Function()
1708 enddef
1709 defcompile
1710 END
1711 CheckScriptSuccess(lines)
1712 assert_false(exists('g:loaded_postponed'))
1713 CheckScriptSuccess(lines + ['Tryit()'])
1714 assert_equal('true', g:loaded_postponed)
1715
1716 unlet g:loaded_postponed
1717 delete('Xdir', 'rf')
1718 &rtp = save_rtp
1719enddef
1720
Bram Moolenaar3e4fa3d2022-01-13 22:05:09 +00001721def Test_import_autoload_override()
1722 mkdir('Xdir/autoload', 'p')
1723 var save_rtp = &rtp
1724 exe 'set rtp^=' .. getcwd() .. '/Xdir'
1725 test_override('autoload', 1)
1726
1727 var lines =<< trim END
Bram Moolenaarfd218c82022-01-18 16:26:24 +00001728 vim9script
Bram Moolenaar3e4fa3d2022-01-13 22:05:09 +00001729
1730 g:loaded_override = 'true'
1731 export var variable = 'bla'
1732 export def Function(): string
1733 return 'bla'
1734 enddef
1735 END
1736 writefile(lines, 'Xdir/autoload/override.vim')
1737
1738 lines =<< trim END
1739 vim9script
1740
1741 import autoload 'override.vim'
1742 assert_equal('true', g:loaded_override)
1743
1744 def Tryit()
1745 echo override.doesNotExist
1746 enddef
1747 defcompile
1748 END
1749 CheckScriptFailure(lines, 'E1048: Item not found in script: doesNotExist', 1)
1750
1751 test_override('autoload', 0)
1752 unlet g:loaded_override
1753 delete('Xdir', 'rf')
1754 &rtp = save_rtp
1755enddef
1756
Bram Moolenaar19db9e62022-01-11 11:58:19 +00001757def Test_autoload_mapping()
1758 mkdir('Xdir/autoload', 'p')
1759 var save_rtp = &rtp
1760 exe 'set rtp^=' .. getcwd() .. '/Xdir'
1761
1762 var lines =<< trim END
Bram Moolenaarfd218c82022-01-18 16:26:24 +00001763 vim9script
Bram Moolenaar19db9e62022-01-11 11:58:19 +00001764
1765 g:toggle_loaded = 'yes'
1766
1767 export def Toggle(): string
1768 return ":g:toggle_called = 'yes'\<CR>"
1769 enddef
Bram Moolenaare32c3c42022-01-15 18:26:04 +00001770 export def Doit()
1771 g:doit_called = 'yes'
1772 enddef
Bram Moolenaar19db9e62022-01-11 11:58:19 +00001773 END
1774 writefile(lines, 'Xdir/autoload/toggle.vim')
1775
1776 lines =<< trim END
1777 vim9script
1778
1779 import autoload 'toggle.vim'
1780
1781 nnoremap <silent> <expr> tt toggle.Toggle()
Bram Moolenaare32c3c42022-01-15 18:26:04 +00001782 nnoremap <silent> xx <ScriptCmd>toggle.Doit()<CR>
1783 nnoremap <silent> yy <Cmd>toggle.Doit()<CR>
Bram Moolenaar19db9e62022-01-11 11:58:19 +00001784 END
1785 CheckScriptSuccess(lines)
1786 assert_false(exists("g:toggle_loaded"))
1787 assert_false(exists("g:toggle_called"))
Bram Moolenaar6079da72022-01-18 14:16:59 +00001788 assert_match('\d A: \f*[/\\]toggle.vim', execute('scriptnames'))
Bram Moolenaar19db9e62022-01-11 11:58:19 +00001789
1790 feedkeys("tt", 'xt')
1791 assert_equal('yes', g:toggle_loaded)
1792 assert_equal('yes', g:toggle_called)
Bram Moolenaar6079da72022-01-18 14:16:59 +00001793 assert_match('\d: \f*[/\\]toggle.vim', execute('scriptnames'))
Bram Moolenaar19db9e62022-01-11 11:58:19 +00001794
Bram Moolenaare32c3c42022-01-15 18:26:04 +00001795 feedkeys("xx", 'xt')
1796 assert_equal('yes', g:doit_called)
1797
1798 assert_fails('call feedkeys("yy", "xt")', 'E121: Undefined variable: toggle')
1799
Bram Moolenaar19db9e62022-01-11 11:58:19 +00001800 nunmap tt
Bram Moolenaare32c3c42022-01-15 18:26:04 +00001801 nunmap xx
1802 nunmap yy
Bram Moolenaar19db9e62022-01-11 11:58:19 +00001803 unlet g:toggle_loaded
1804 unlet g:toggle_called
1805 delete('Xdir', 'rf')
1806 &rtp = save_rtp
1807enddef
1808
Bram Moolenaar160aa862022-01-10 21:29:57 +00001809def Test_vim9script_autoload_fails()
1810 var lines =<< trim END
1811 vim9script autoload
1812 var n = 0
1813 END
Bram Moolenaarfd218c82022-01-18 16:26:24 +00001814 CheckScriptFailure(lines, 'E475: Invalid argument: autoload')
1815
1816 lines =<< trim END
1817 vim9script noclear noclear
1818 var n = 0
1819 END
1820 CheckScriptFailure(lines, 'E983: Duplicate argument: noclear')
Bram Moolenaar160aa862022-01-10 21:29:57 +00001821enddef
1822
1823def Test_import_autoload_fails()
1824 var lines =<< trim END
1825 vim9script
1826 import autoload autoload 'prefixed.vim'
1827 END
1828 CheckScriptFailure(lines, 'E121: Undefined variable: autoload')
1829
1830 lines =<< trim END
1831 vim9script
Bram Moolenaar1836d612022-01-18 13:14:47 +00001832 import autoload './doesNotExist.vim'
Bram Moolenaar160aa862022-01-10 21:29:57 +00001833 END
1834 CheckScriptFailure(lines, 'E1264:')
Bram Moolenaar1836d612022-01-18 13:14:47 +00001835
1836 lines =<< trim END
1837 vim9script
1838 import autoload '/dir/doesNotExist.vim'
1839 END
1840 CheckScriptFailure(lines, 'E1264:')
1841
1842 lines =<< trim END
1843 vim9script
1844 import autoload 'doesNotExist.vim'
1845 END
1846 CheckScriptFailure(lines, 'E1053: Could not import "doesNotExist.vim"')
Bram Moolenaar160aa862022-01-10 21:29:57 +00001847enddef
1848
1849" test disassembling an auto-loaded function starting with "debug"
1850def Test_vim9_autoload_disass()
1851 mkdir('Xdir/autoload', 'p')
1852 var save_rtp = &rtp
1853 exe 'set rtp^=' .. getcwd() .. '/Xdir'
1854
1855 var lines =<< trim END
1856 vim9script
1857 def debugit#test(): string
1858 return 'debug'
1859 enddef
1860 END
1861 writefile(lines, 'Xdir/autoload/debugit.vim')
1862
1863 lines =<< trim END
1864 vim9script
1865 def profileit#test(): string
1866 return 'profile'
1867 enddef
1868 END
1869 writefile(lines, 'Xdir/autoload/profileit.vim')
1870
1871 lines =<< trim END
1872 vim9script
1873 assert_equal('debug', debugit#test())
1874 disass debugit#test
1875 assert_equal('profile', profileit#test())
1876 disass profileit#test
1877 END
1878 CheckScriptSuccess(lines)
1879
1880 delete('Xdir', 'rf')
1881 &rtp = save_rtp
1882enddef
1883
1884" test using a vim9script that is auto-loaded from an autocmd
1885def Test_vim9_aucmd_autoload()
1886 var lines =<< trim END
1887 vim9script
1888 def foo#test()
1889 echomsg getreg('"')
1890 enddef
1891 END
1892
1893 mkdir('Xdir/autoload', 'p')
1894 writefile(lines, 'Xdir/autoload/foo.vim')
1895 var save_rtp = &rtp
1896 exe 'set rtp^=' .. getcwd() .. '/Xdir'
1897 augroup test
1898 autocmd TextYankPost * call foo#test()
1899 augroup END
1900
1901 normal Y
1902
1903 augroup test
1904 autocmd!
1905 augroup END
1906 delete('Xdir', 'rf')
1907 &rtp = save_rtp
1908enddef
1909
Bram Moolenaar3049fcf2022-01-13 19:25:50 +00001910" test using a autoloaded file that is case sensitive
1911def Test_vim9_autoload_case_sensitive()
1912 var lines =<< trim END
Bram Moolenaarfd218c82022-01-18 16:26:24 +00001913 vim9script
Bram Moolenaar3049fcf2022-01-13 19:25:50 +00001914 export def CaseSensitive(): string
1915 return 'done'
1916 enddef
1917 END
1918
1919 mkdir('Xdir/autoload', 'p')
1920 writefile(lines, 'Xdir/autoload/CaseSensitive.vim')
1921 var save_rtp = &rtp
1922 exe 'set rtp^=' .. getcwd() .. '/Xdir'
1923
1924 lines =<< trim END
1925 vim9script
1926 import autoload 'CaseSensitive.vim'
1927 assert_equal('done', CaseSensitive.CaseSensitive())
1928 END
1929 CheckScriptSuccess(lines)
1930
Bram Moolenaarbfac4092022-01-16 11:12:12 +00001931 if !has('fname_case')
1932 lines =<< trim END
1933 vim9script
1934 import autoload 'CaseSensitive.vim'
1935 import autoload 'casesensitive.vim'
1936 END
1937 CheckScriptFailure(lines, 'E1262:')
1938 endif
1939
Bram Moolenaar3049fcf2022-01-13 19:25:50 +00001940 delete('Xdir', 'rf')
1941 &rtp = save_rtp
1942enddef
1943
Bram Moolenaar160aa862022-01-10 21:29:57 +00001944" This was causing a crash because suppress_errthrow wasn't reset.
1945def Test_vim9_autoload_error()
1946 var lines =<< trim END
1947 vim9script
1948 def crash#func()
1949 try
1950 for x in List()
1951 endfor
1952 catch
1953 endtry
1954 g:ok = true
1955 enddef
1956 fu List()
1957 invalid
1958 endfu
1959 try
1960 alsoinvalid
1961 catch /wontmatch/
1962 endtry
1963 END
1964 call mkdir('Xruntime/autoload', 'p')
1965 call writefile(lines, 'Xruntime/autoload/crash.vim')
1966
1967 # run in a separate Vim to avoid the side effects of assert_fails()
1968 lines =<< trim END
1969 exe 'set rtp^=' .. getcwd() .. '/Xruntime'
1970 call crash#func()
1971 call writefile(['ok'], 'Xdidit')
1972 qall!
1973 END
1974 writefile(lines, 'Xscript')
1975 RunVim([], [], '-S Xscript')
1976 assert_equal(['ok'], readfile('Xdidit'))
1977
1978 delete('Xdidit')
1979 delete('Xscript')
1980 delete('Xruntime', 'rf')
1981
1982 lines =<< trim END
1983 vim9script
1984 var foo#bar = 'asdf'
1985 END
1986 CheckScriptFailure(lines, 'E461: Illegal variable name: foo#bar', 2)
1987enddef
1988
Bram Moolenaard8448622022-01-07 21:39:52 +00001989
1990" vim: ts=8 sw=2 sts=2 expandtab tw=80 fdm=marker