1" Test using builtin functions in the Vim9 script language.
2
3source check.vim
4source vim9.vim
5
6" Test for passing too many or too few arguments to builtin functions
7func Test_internalfunc_arg_error()
8  let l =<< trim END
9    def! FArgErr(): float
10      return ceil(1.1, 2)
11    enddef
12    defcompile
13  END
14  call writefile(l, 'Xinvalidarg')
15  call assert_fails('so Xinvalidarg', 'E118:', '', 1, 'FArgErr')
16  let l =<< trim END
17    def! FArgErr(): float
18      return ceil()
19    enddef
20    defcompile
21  END
22  call writefile(l, 'Xinvalidarg')
23  call assert_fails('so Xinvalidarg', 'E119:', '', 1, 'FArgErr')
24  call delete('Xinvalidarg')
25endfunc
26
27" Test for builtin functions returning different types
28func Test_InternalFuncRetType()
29  let lines =<< trim END
30    def RetFloat(): float
31      return ceil(1.456)
32    enddef
33
34    def RetListAny(): list<any>
35      return items({k: 'v'})
36    enddef
37
38    def RetListString(): list<string>
39      return split('a:b:c', ':')
40    enddef
41
42    def RetListDictAny(): list<dict<any>>
43      return getbufinfo()
44    enddef
45
46    def RetDictNumber(): dict<number>
47      return wordcount()
48    enddef
49
50    def RetDictString(): dict<string>
51      return environ()
52    enddef
53  END
54  call writefile(lines, 'Xscript')
55  source Xscript
56
57  call RetFloat()->assert_equal(2.0)
58  call RetListAny()->assert_equal([['k', 'v']])
59  call RetListString()->assert_equal(['a', 'b', 'c'])
60  call RetListDictAny()->assert_notequal([])
61  call RetDictNumber()->assert_notequal({})
62  call RetDictString()->assert_notequal({})
63  call delete('Xscript')
64endfunc
65
66def Test_abs()
67  assert_equal(0, abs(0))
68  assert_equal(2, abs(-2))
69  assert_equal(3, abs(3))
70  CheckDefFailure(['abs("text")'], 'E1013: Argument 1: type mismatch, expected number but got string', 1)
71  if has('float')
72    assert_equal(0, abs(0))
73    assert_equal(2.0, abs(-2.0))
74    assert_equal(3.0, abs(3.0))
75  endif
76enddef
77
78def Test_add_list()
79  var l: list<number>  # defaults to empty list
80  add(l, 9)
81  assert_equal([9], l)
82
83  var lines =<< trim END
84      var l: list<number>
85      add(l, "x")
86  END
87  CheckDefFailure(lines, 'E1012:', 2)
88
89  lines =<< trim END
90      var l: list<number> = test_null_list()
91      add(l, 123)
92  END
93  CheckDefExecFailure(lines, 'E1130:', 2)
94enddef
95
96def Test_add_blob()
97  var b1: blob = 0z12
98  add(b1, 0x34)
99  assert_equal(0z1234, b1)
100
101  var b2: blob # defaults to empty blob
102  add(b2, 0x67)
103  assert_equal(0z67, b2)
104
105  var lines =<< trim END
106      var b: blob
107      add(b, "x")
108  END
109  CheckDefFailure(lines, 'E1012:', 2)
110
111  lines =<< trim END
112      var b: blob = test_null_blob()
113      add(b, 123)
114  END
115  CheckDefExecFailure(lines, 'E1131:', 2)
116enddef
117
118def Test_append()
119  new
120  setline(1, range(3))
121  var res1: number = append(1, 'one')
122  assert_equal(0, res1)
123  var res2: bool = append(3, 'two')
124  assert_equal(false, res2)
125  assert_equal(['0', 'one', '1', 'two', '2'], getline(1, 6))
126enddef
127
128def Test_buflisted()
129  var res: bool = buflisted('asdf')
130  assert_equal(false, res)
131enddef
132
133def Test_bufname()
134  split SomeFile
135  bufname('%')->assert_equal('SomeFile')
136  edit OtherFile
137  bufname('#')->assert_equal('SomeFile')
138  close
139enddef
140
141def Test_bufnr()
142  var buf = bufnr()
143  bufnr('%')->assert_equal(buf)
144
145  buf = bufnr('Xdummy', true)
146  buf->assert_notequal(-1)
147  exe 'bwipe! ' .. buf
148enddef
149
150def Test_bufwinid()
151  var origwin = win_getid()
152  below split SomeFile
153  var SomeFileID = win_getid()
154  below split OtherFile
155  below split SomeFile
156  bufwinid('SomeFile')->assert_equal(SomeFileID)
157
158  win_gotoid(origwin)
159  only
160  bwipe SomeFile
161  bwipe OtherFile
162enddef
163
164def Test_call_call()
165  var l = [3, 2, 1]
166  call('reverse', [l])
167  l->assert_equal([1, 2, 3])
168enddef
169
170def Test_char2nr()
171  char2nr('あ', true)->assert_equal(12354)
172enddef
173
174def Test_col()
175  new
176  setline(1, 'asdf')
177  col([1, '$'])->assert_equal(5)
178enddef
179
180def Test_copy_return_type()
181  var l = copy([1, 2, 3])
182  var res = 0
183  for n in l
184    res += n
185  endfor
186  res->assert_equal(6)
187
188  var dl = deepcopy([1, 2, 3])
189  res = 0
190  for n in dl
191    res += n
192  endfor
193  res->assert_equal(6)
194
195  dl = deepcopy([1, 2, 3], true)
196enddef
197
198def Test_count()
199  count('ABC ABC ABC', 'b', true)->assert_equal(3)
200  count('ABC ABC ABC', 'b', false)->assert_equal(0)
201enddef
202
203def Test_cursor()
204  new
205  setline(1, range(4))
206  cursor(2, 1)
207  assert_equal(2, getcurpos()[1])
208  cursor('$', 1)
209  assert_equal(4, getcurpos()[1])
210
211  var lines =<< trim END
212    cursor('2', 1)
213  END
214  CheckDefExecAndScriptFailure(lines, 'E475:')
215enddef
216
217def Test_delete()
218  var res: bool = delete('doesnotexist')
219  assert_equal(true, res)
220enddef
221
222def Test_executable()
223  assert_false(executable(""))
224  assert_false(executable(test_null_string()))
225
226  CheckDefExecFailure(['echo executable(123)'], 'E928:')
227  CheckDefExecFailure(['echo executable(true)'], 'E928:')
228enddef
229
230def Test_exepath()
231  CheckDefExecFailure(['echo exepath(true)'], 'E928:')
232  CheckDefExecFailure(['echo exepath(v:null)'], 'E928:')
233  CheckDefExecFailure(['echo exepath("")'], 'E1142:')
234enddef
235
236def Test_expand()
237  split SomeFile
238  expand('%', true, true)->assert_equal(['SomeFile'])
239  close
240enddef
241
242def Test_extend_arg_types()
243  g:number_one = 1
244  g:string_keep = 'keep'
245  var lines =<< trim END
246      assert_equal([1, 2, 3], extend([1, 2], [3]))
247      assert_equal([3, 1, 2], extend([1, 2], [3], 0))
248      assert_equal([1, 3, 2], extend([1, 2], [3], 1))
249      assert_equal([1, 3, 2], extend([1, 2], [3], g:number_one))
250
251      assert_equal({a: 1, b: 2, c: 3}, extend({a: 1, b: 2}, {c: 3}))
252      assert_equal({a: 1, b: 4}, extend({a: 1, b: 2}, {b: 4}))
253      assert_equal({a: 1, b: 2}, extend({a: 1, b: 2}, {b: 4}, 'keep'))
254      assert_equal({a: 1, b: 2}, extend({a: 1, b: 2}, {b: 4}, g:string_keep))
255
256      var res: list<dict<any>>
257      extend(res, mapnew([1, 2], (_, v) => ({})))
258      assert_equal([{}, {}], res)
259  END
260  CheckDefAndScriptSuccess(lines)
261
262  CheckDefFailure(['extend([1, 2], 3)'], 'E1013: Argument 2: type mismatch, expected list<number> but got number')
263  CheckDefFailure(['extend([1, 2], ["x"])'], 'E1013: Argument 2: type mismatch, expected list<number> but got list<string>')
264  CheckDefFailure(['extend([1, 2], [3], "x")'], 'E1013: Argument 3: type mismatch, expected number but got string')
265
266  CheckDefFailure(['extend({a: 1}, 42)'], 'E1013: Argument 2: type mismatch, expected dict<number> but got number')
267  CheckDefFailure(['extend({a: 1}, {b: "x"})'], 'E1013: Argument 2: type mismatch, expected dict<number> but got dict<string>')
268  CheckDefFailure(['extend({a: 1}, {b: 2}, 1)'], 'E1013: Argument 3: type mismatch, expected string but got number')
269
270  CheckDefFailure(['extend([1], ["b"])'], 'E1013: Argument 2: type mismatch, expected list<number> but got list<string>')
271  CheckDefExecFailure(['extend([1], ["b", 1])'], 'E1013: Argument 2: type mismatch, expected list<number> but got list<any>')
272enddef
273
274def Test_extendnew()
275  assert_equal([1, 2, 'a'], extendnew([1, 2], ['a']))
276  assert_equal({one: 1, two: 'a'}, extendnew({one: 1}, {two: 'a'}))
277
278  CheckDefFailure(['extendnew({a: 1}, 42)'], 'E1013: Argument 2: type mismatch, expected dict<number> but got number')
279  CheckDefFailure(['extendnew({a: 1}, [42])'], 'E1013: Argument 2: type mismatch, expected dict<number> but got list<number>')
280  CheckDefFailure(['extendnew([1, 2], "x")'], 'E1013: Argument 2: type mismatch, expected list<number> but got string')
281  CheckDefFailure(['extendnew([1, 2], {x: 1})'], 'E1013: Argument 2: type mismatch, expected list<number> but got dict<number>')
282enddef
283
284def Test_extend_return_type()
285  var l = extend([1, 2], [3])
286  var res = 0
287  for n in l
288    res += n
289  endfor
290  res->assert_equal(6)
291enddef
292
293func g:ExtendDict(d)
294  call extend(a:d, #{xx: 'x'})
295endfunc
296
297def Test_extend_dict_item_type()
298  var lines =<< trim END
299       var d: dict<number> = {a: 1}
300       extend(d, {b: 2})
301  END
302  CheckDefAndScriptSuccess(lines)
303
304  lines =<< trim END
305       var d: dict<number> = {a: 1}
306       extend(d, {b: 'x'})
307  END
308  CheckDefAndScriptFailure(lines, 'E1013: Argument 2: type mismatch, expected dict<number> but got dict<string>', 2)
309
310  lines =<< trim END
311       var d: dict<number> = {a: 1}
312       g:ExtendDict(d)
313  END
314  CheckDefExecFailure(lines, 'E1012: Type mismatch; expected number but got string', 0)
315  CheckScriptFailure(['vim9script'] + lines, 'E1012:', 1)
316enddef
317
318func g:ExtendList(l)
319  call extend(a:l, ['x'])
320endfunc
321
322def Test_extend_list_item_type()
323  var lines =<< trim END
324       var l: list<number> = [1]
325       extend(l, [2])
326  END
327  CheckDefAndScriptSuccess(lines)
328
329  lines =<< trim END
330       var l: list<number> = [1]
331       extend(l, ['x'])
332  END
333  CheckDefAndScriptFailure(lines, 'E1013: Argument 2: type mismatch, expected list<number> but got list<string>', 2)
334
335  lines =<< trim END
336       var l: list<number> = [1]
337       g:ExtendList(l)
338  END
339  CheckDefExecFailure(lines, 'E1012: Type mismatch; expected number but got string', 0)
340  CheckScriptFailure(['vim9script'] + lines, 'E1012:', 1)
341enddef
342
343def Test_job_info_return_type()
344  if has('job')
345    job_start(&shell)
346    var jobs = job_info()
347    assert_equal('list<job>', typename(jobs))
348    assert_equal('dict<any>', typename(job_info(jobs[0])))
349    job_stop(jobs[0])
350  endif
351enddef
352
353def Test_filereadable()
354  assert_false(filereadable(""))
355  assert_false(filereadable(test_null_string()))
356
357  CheckDefExecFailure(['echo filereadable(123)'], 'E928:')
358  CheckDefExecFailure(['echo filereadable(true)'], 'E928:')
359enddef
360
361def Test_filewritable()
362  assert_false(filewritable(""))
363  assert_false(filewritable(test_null_string()))
364
365  CheckDefExecFailure(['echo filewritable(123)'], 'E928:')
366  CheckDefExecFailure(['echo filewritable(true)'], 'E928:')
367enddef
368
369def Test_finddir()
370  CheckDefExecFailure(['echo finddir(true)'], 'E928:')
371  CheckDefExecFailure(['echo finddir(v:null)'], 'E928:')
372  CheckDefExecFailure(['echo finddir("")'], 'E1142:')
373enddef
374
375def Test_findfile()
376  CheckDefExecFailure(['echo findfile(true)'], 'E928:')
377  CheckDefExecFailure(['echo findfile(v:null)'], 'E928:')
378  CheckDefExecFailure(['echo findfile("")'], 'E1142:')
379enddef
380
381def Test_flattennew()
382  var lines =<< trim END
383      var l = [1, [2, [3, 4]], 5]
384      call assert_equal([1, 2, 3, 4, 5], flattennew(l))
385      call assert_equal([1, [2, [3, 4]], 5], l)
386
387      call assert_equal([1, 2, [3, 4], 5], flattennew(l, 1))
388      call assert_equal([1, [2, [3, 4]], 5], l)
389  END
390  CheckDefAndScriptSuccess(lines)
391
392  lines =<< trim END
393      echo flatten([1, 2, 3])
394  END
395  CheckDefAndScriptFailure(lines, 'E1158:')
396enddef
397
398def Test_fnamemodify()
399  CheckDefSuccess(['echo fnamemodify(test_null_string(), ":p")'])
400  CheckDefSuccess(['echo fnamemodify("", ":p")'])
401  CheckDefSuccess(['echo fnamemodify("file", test_null_string())'])
402  CheckDefSuccess(['echo fnamemodify("file", "")'])
403
404  CheckDefExecFailure(['echo fnamemodify(true, ":p")'], 'E928:')
405  CheckDefExecFailure(['echo fnamemodify(v:null, ":p")'], 'E928:')
406  CheckDefExecFailure(['echo fnamemodify("file", true)'], 'E928:')
407enddef
408
409def Wrong_dict_key_type(items: list<number>): list<number>
410  return filter(items, (_, val) => get({[val]: 1}, 'x'))
411enddef
412
413def Test_filter_wrong_dict_key_type()
414  assert_fails('Wrong_dict_key_type([1, v:null, 3])', 'E1013:')
415enddef
416
417def Test_filter_return_type()
418  var l = filter([1, 2, 3], () => 1)
419  var res = 0
420  for n in l
421    res += n
422  endfor
423  res->assert_equal(6)
424enddef
425
426def Test_filter_missing_argument()
427  var dict = {aa: [1], ab: [2], ac: [3], de: [4]}
428  var res = dict->filter((k) => k =~ 'a' && k !~ 'b')
429  res->assert_equal({aa: [1], ac: [3]})
430enddef
431
432def Test_garbagecollect()
433  garbagecollect(true)
434enddef
435
436def Test_getbufinfo()
437  var bufinfo = getbufinfo(bufnr())
438  getbufinfo('%')->assert_equal(bufinfo)
439
440  edit Xtestfile1
441  hide edit Xtestfile2
442  hide enew
443  getbufinfo({bufloaded: true, buflisted: true, bufmodified: false})
444      ->len()->assert_equal(3)
445  bwipe Xtestfile1 Xtestfile2
446enddef
447
448def Test_getbufline()
449  e SomeFile
450  var buf = bufnr()
451  e #
452  var lines = ['aaa', 'bbb', 'ccc']
453  setbufline(buf, 1, lines)
454  getbufline('#', 1, '$')->assert_equal(lines)
455  getbufline(-1, '$', '$')->assert_equal([])
456  getbufline(-1, 1, '$')->assert_equal([])
457
458  bwipe!
459enddef
460
461def Test_getchangelist()
462  new
463  setline(1, 'some text')
464  var changelist = bufnr()->getchangelist()
465  getchangelist('%')->assert_equal(changelist)
466  bwipe!
467enddef
468
469def Test_getchar()
470  while getchar(0)
471  endwhile
472  getchar(true)->assert_equal(0)
473enddef
474
475def Test_getcompletion()
476  set wildignore=*.vim,*~
477  var l = getcompletion('run', 'file', true)
478  l->assert_equal([])
479  set wildignore&
480enddef
481
482def Test_getloclist_return_type()
483  var l = getloclist(1)
484  l->assert_equal([])
485
486  var d = getloclist(1, {items: 0})
487  d->assert_equal({items: []})
488enddef
489
490def Test_getfperm()
491  assert_equal('', getfperm(""))
492  assert_equal('', getfperm(test_null_string()))
493
494  CheckDefExecFailure(['echo getfperm(true)'], 'E928:')
495  CheckDefExecFailure(['echo getfperm(v:null)'], 'E928:')
496enddef
497
498def Test_getfsize()
499  assert_equal(-1, getfsize(""))
500  assert_equal(-1, getfsize(test_null_string()))
501
502  CheckDefExecFailure(['echo getfsize(true)'], 'E928:')
503  CheckDefExecFailure(['echo getfsize(v:null)'], 'E928:')
504enddef
505
506def Test_getftime()
507  assert_equal(-1, getftime(""))
508  assert_equal(-1, getftime(test_null_string()))
509
510  CheckDefExecFailure(['echo getftime(true)'], 'E928:')
511  CheckDefExecFailure(['echo getftime(v:null)'], 'E928:')
512enddef
513
514def Test_getftype()
515  assert_equal('', getftype(""))
516  assert_equal('', getftype(test_null_string()))
517
518  CheckDefExecFailure(['echo getftype(true)'], 'E928:')
519  CheckDefExecFailure(['echo getftype(v:null)'], 'E928:')
520enddef
521
522def Test_getqflist_return_type()
523  var l = getqflist()
524  l->assert_equal([])
525
526  var d = getqflist({items: 0})
527  d->assert_equal({items: []})
528enddef
529
530def Test_getreg()
531  var lines = ['aaa', 'bbb', 'ccc']
532  setreg('a', lines)
533  getreg('a', true, true)->assert_equal(lines)
534  assert_fails('getreg("ab")', 'E1162:')
535enddef
536
537def Test_getreg_return_type()
538  var s1: string = getreg('"')
539  var s2: string = getreg('"', 1)
540  var s3: list<string> = getreg('"', 1, 1)
541enddef
542
543def Test_getreginfo()
544  var text = 'abc'
545  setreg('a', text)
546  getreginfo('a')->assert_equal({regcontents: [text], regtype: 'v', isunnamed: false})
547  assert_fails('getreginfo("ab")', 'E1162:')
548enddef
549
550def Test_getregtype()
551  var lines = ['aaa', 'bbb', 'ccc']
552  setreg('a', lines)
553  getregtype('a')->assert_equal('V')
554  assert_fails('getregtype("ab")', 'E1162:')
555enddef
556
557def Test_glob()
558  glob('runtest.vim', true, true, true)->assert_equal(['runtest.vim'])
559enddef
560
561def Test_globpath()
562  globpath('.', 'runtest.vim', true, true, true)->assert_equal(['./runtest.vim'])
563enddef
564
565def Test_has()
566  has('eval', true)->assert_equal(1)
567enddef
568
569def Test_hasmapto()
570  hasmapto('foobar', 'i', true)->assert_equal(0)
571  iabbrev foo foobar
572  hasmapto('foobar', 'i', true)->assert_equal(1)
573  iunabbrev foo
574enddef
575
576def Test_index()
577  index(['a', 'b', 'a', 'B'], 'b', 2, true)->assert_equal(3)
578enddef
579
580let s:number_one = 1
581let s:number_two = 2
582let s:string_keep = 'keep'
583
584def Test_insert()
585  var l = insert([2, 1], 3)
586  var res = 0
587  for n in l
588    res += n
589  endfor
590  res->assert_equal(6)
591
592  assert_equal([1, 2, 3], insert([2, 3], 1))
593  assert_equal([1, 2, 3], insert([2, 3], s:number_one))
594  assert_equal([1, 2, 3], insert([1, 2], 3, 2))
595  assert_equal([1, 2, 3], insert([1, 2], 3, s:number_two))
596  assert_equal(['a', 'b', 'c'], insert(['b', 'c'], 'a'))
597  assert_equal(0z1234, insert(0z34, 0x12))
598
599  CheckDefFailure(['insert([2, 3], "a")'], 'E1013: Argument 2: type mismatch, expected number but got string', 1)
600  CheckDefFailure(['insert([2, 3], 1, "x")'], 'E1013: Argument 3: type mismatch, expected number but got string', 1)
601enddef
602
603def Test_keys_return_type()
604  const var: list<string> = {a: 1, b: 2}->keys()
605  var->assert_equal(['a', 'b'])
606enddef
607
608def Test_list2str_str2list_utf8()
609  var s = "\u3042\u3044"
610  var l = [0x3042, 0x3044]
611  str2list(s, true)->assert_equal(l)
612  list2str(l, true)->assert_equal(s)
613enddef
614
615def SID(): number
616  return expand('<SID>')
617          ->matchstr('<SNR>\zs\d\+\ze_$')
618          ->str2nr()
619enddef
620
621def Test_map_function_arg()
622  var lines =<< trim END
623      def MapOne(i: number, v: string): string
624        return i .. ':' .. v
625      enddef
626      var l = ['a', 'b', 'c']
627      map(l, MapOne)
628      assert_equal(['0:a', '1:b', '2:c'], l)
629  END
630  CheckDefAndScriptSuccess(lines)
631enddef
632
633def Test_map_item_type()
634  var lines =<< trim END
635      var l = ['a', 'b', 'c']
636      map(l, (k, v) => k .. '/' .. v )
637      assert_equal(['0/a', '1/b', '2/c'], l)
638  END
639  CheckDefAndScriptSuccess(lines)
640
641  lines =<< trim END
642    var l: list<number> = [0]
643    echo map(l, (_, v) => [])
644  END
645  CheckDefExecAndScriptFailure(lines, 'E1012: Type mismatch; expected number but got list<unknown>', 2)
646
647  lines =<< trim END
648    var l: list<number> = range(2)
649    echo map(l, (_, v) => [])
650  END
651  CheckDefExecAndScriptFailure(lines, 'E1012: Type mismatch; expected number but got list<unknown>', 2)
652
653  lines =<< trim END
654    var d: dict<number> = {key: 0}
655    echo map(d, (_, v) => [])
656  END
657  CheckDefExecAndScriptFailure(lines, 'E1012: Type mismatch; expected number but got list<unknown>', 2)
658enddef
659
660def Test_maparg()
661  var lnum = str2nr(expand('<sflnum>'))
662  map foo bar
663  maparg('foo', '', false, true)->assert_equal({
664        lnum: lnum + 1,
665        script: 0,
666        mode: ' ',
667        silent: 0,
668        noremap: 0,
669        lhs: 'foo',
670        lhsraw: 'foo',
671        nowait: 0,
672        expr: 0,
673        sid: SID(),
674        rhs: 'bar',
675        buffer: 0})
676  unmap foo
677enddef
678
679def Test_mapcheck()
680  iabbrev foo foobar
681  mapcheck('foo', 'i', true)->assert_equal('foobar')
682  iunabbrev foo
683enddef
684
685def Test_maparg_mapset()
686  nnoremap <F3> :echo "hit F3"<CR>
687  var mapsave = maparg('<F3>', 'n', false, true)
688  mapset('n', false, mapsave)
689
690  nunmap <F3>
691enddef
692
693def Test_map_failure()
694  CheckFeature job
695
696  var lines =<< trim END
697      vim9script
698      writefile([], 'Xtmpfile')
699      silent e Xtmpfile
700      var d = {[bufnr('%')]: {a: 0}}
701      au BufReadPost * Func()
702      def Func()
703          if d->has_key('')
704          endif
705          eval d[expand('<abuf>')]->mapnew((_, v: dict<job>) => 0)
706      enddef
707      e
708  END
709  CheckScriptFailure(lines, 'E1013:')
710  au! BufReadPost
711  delete('Xtmpfile')
712enddef
713
714def Test_max()
715  g:flag = true
716  var l1: list<number> = g:flag
717          ? [1, max([2, 3])]
718          : [4, 5]
719  assert_equal([1, 3], l1)
720
721  g:flag = false
722  var l2: list<number> = g:flag
723          ? [1, max([2, 3])]
724          : [4, 5]
725  assert_equal([4, 5], l2)
726enddef
727
728def Test_min()
729  g:flag = true
730  var l1: list<number> = g:flag
731          ? [1, min([2, 3])]
732          : [4, 5]
733  assert_equal([1, 2], l1)
734
735  g:flag = false
736  var l2: list<number> = g:flag
737          ? [1, min([2, 3])]
738          : [4, 5]
739  assert_equal([4, 5], l2)
740enddef
741
742def Test_nr2char()
743  nr2char(97, true)->assert_equal('a')
744enddef
745
746def Test_readdir()
747   eval expand('sautest')->readdir((e) => e[0] !=# '.')
748   eval expand('sautest')->readdirex((e) => e.name[0] !=# '.')
749enddef
750
751def Test_readblob()
752  var blob = 0z12341234
753  writefile(blob, 'Xreadblob')
754  var read: blob = readblob('Xreadblob')
755  assert_equal(blob, read)
756
757  var lines =<< trim END
758      var read: list<string> = readblob('Xreadblob')
759  END
760  CheckDefAndScriptFailure(lines, 'E1012: Type mismatch; expected list<string> but got blob', 1)
761  delete('Xreadblob')
762enddef
763
764def Test_readfile()
765  var text = ['aaa', 'bbb', 'ccc']
766  writefile(text, 'Xreadfile')
767  var read: list<string> = readfile('Xreadfile')
768  assert_equal(text, read)
769
770  var lines =<< trim END
771      var read: dict<string> = readfile('Xreadfile')
772  END
773  CheckDefAndScriptFailure(lines, 'E1012: Type mismatch; expected dict<string> but got list<string>', 1)
774  delete('Xreadfile')
775enddef
776
777def Test_remove_return_type()
778  var l = remove({one: [1, 2], two: [3, 4]}, 'one')
779  var res = 0
780  for n in l
781    res += n
782  endfor
783  res->assert_equal(3)
784enddef
785
786def Test_reverse_return_type()
787  var l = reverse([1, 2, 3])
788  var res = 0
789  for n in l
790    res += n
791  endfor
792  res->assert_equal(6)
793enddef
794
795def Test_search()
796  new
797  setline(1, ['foo', 'bar'])
798  var val = 0
799  # skip expr returns boolean
800  search('bar', 'W', 0, 0, () => val == 1)->assert_equal(2)
801  :1
802  search('bar', 'W', 0, 0, () => val == 0)->assert_equal(0)
803  # skip expr returns number, only 0 and 1 are accepted
804  :1
805  search('bar', 'W', 0, 0, () => 0)->assert_equal(2)
806  :1
807  search('bar', 'W', 0, 0, () => 1)->assert_equal(0)
808  assert_fails("search('bar', '', 0, 0, () => -1)", 'E1023:')
809  assert_fails("search('bar', '', 0, 0, () => -1)", 'E1023:')
810enddef
811
812def Test_searchcount()
813  new
814  setline(1, "foo bar")
815  :/foo
816  searchcount({recompute: true})
817      ->assert_equal({
818          exact_match: 1,
819          current: 1,
820          total: 1,
821          maxcount: 99,
822          incomplete: 0})
823  bwipe!
824enddef
825
826def Test_set_get_bufline()
827  # similar to Test_setbufline_getbufline()
828  var lines =<< trim END
829      new
830      var b = bufnr('%')
831      hide
832      assert_equal(0, setbufline(b, 1, ['foo', 'bar']))
833      assert_equal(['foo'], getbufline(b, 1))
834      assert_equal(['bar'], getbufline(b, '$'))
835      assert_equal(['foo', 'bar'], getbufline(b, 1, 2))
836      exe "bd!" b
837      assert_equal([], getbufline(b, 1, 2))
838
839      split Xtest
840      setline(1, ['a', 'b', 'c'])
841      b = bufnr('%')
842      wincmd w
843
844      assert_equal(1, setbufline(b, 5, 'x'))
845      assert_equal(1, setbufline(b, 5, ['x']))
846      assert_equal(1, setbufline(b, 5, []))
847      assert_equal(1, setbufline(b, 5, test_null_list()))
848
849      assert_equal(1, 'x'->setbufline(bufnr('$') + 1, 1))
850      assert_equal(1, ['x']->setbufline(bufnr('$') + 1, 1))
851      assert_equal(1, []->setbufline(bufnr('$') + 1, 1))
852      assert_equal(1, test_null_list()->setbufline(bufnr('$') + 1, 1))
853
854      assert_equal(['a', 'b', 'c'], getbufline(b, 1, '$'))
855
856      assert_equal(0, setbufline(b, 4, ['d', 'e']))
857      assert_equal(['c'], b->getbufline(3))
858      assert_equal(['d'], getbufline(b, 4))
859      assert_equal(['e'], getbufline(b, 5))
860      assert_equal([], getbufline(b, 6))
861      assert_equal([], getbufline(b, 2, 1))
862
863      if has('job')
864        setbufline(b, 2, [function('eval'), {key: 123}, test_null_job()])
865        assert_equal(["function('eval')",
866                        "{'key': 123}",
867                        "no process"],
868                        getbufline(b, 2, 4))
869      endif
870
871      exe 'bwipe! ' .. b
872  END
873  CheckDefAndScriptSuccess(lines)
874enddef
875
876def Test_searchdecl()
877  searchdecl('blah', true, true)->assert_equal(1)
878enddef
879
880def Test_setbufvar()
881  setbufvar(bufnr('%'), '&syntax', 'vim')
882  &syntax->assert_equal('vim')
883  setbufvar(bufnr('%'), '&ts', 16)
884  &ts->assert_equal(16)
885  setbufvar(bufnr('%'), '&ai', true)
886  &ai->assert_equal(true)
887  setbufvar(bufnr('%'), '&ft', 'filetype')
888  &ft->assert_equal('filetype')
889
890  settabwinvar(1, 1, '&syntax', 'vam')
891  &syntax->assert_equal('vam')
892  settabwinvar(1, 1, '&ts', 15)
893  &ts->assert_equal(15)
894  setlocal ts=8
895  settabwinvar(1, 1, '&list', false)
896  &list->assert_equal(false)
897  settabwinvar(1, 1, '&list', true)
898  &list->assert_equal(true)
899  setlocal list&
900
901  setbufvar('%', 'myvar', 123)
902  getbufvar('%', 'myvar')->assert_equal(123)
903enddef
904
905def Test_setloclist()
906  var items = [{filename: '/tmp/file', lnum: 1, valid: true}]
907  var what = {items: items}
908  setqflist([], ' ', what)
909  setloclist(0, [], ' ', what)
910enddef
911
912def Test_setreg()
913  setreg('a', ['aaa', 'bbb', 'ccc'])
914  var reginfo = getreginfo('a')
915  setreg('a', reginfo)
916  getreginfo('a')->assert_equal(reginfo)
917  assert_fails('setreg("ab", 0)', 'E1162:')
918enddef
919
920def Test_slice()
921  assert_equal('12345', slice('012345', 1))
922  assert_equal('123', slice('012345', 1, 4))
923  assert_equal('1234', slice('012345', 1, -1))
924  assert_equal('1', slice('012345', 1, -4))
925  assert_equal('', slice('012345', 1, -5))
926  assert_equal('', slice('012345', 1, -6))
927
928  assert_equal([1, 2, 3, 4, 5], slice(range(6), 1))
929  assert_equal([1, 2, 3], slice(range(6), 1, 4))
930  assert_equal([1, 2, 3, 4], slice(range(6), 1, -1))
931  assert_equal([1], slice(range(6), 1, -4))
932  assert_equal([], slice(range(6), 1, -5))
933  assert_equal([], slice(range(6), 1, -6))
934
935  assert_equal(0z1122334455, slice(0z001122334455, 1))
936  assert_equal(0z112233, slice(0z001122334455, 1, 4))
937  assert_equal(0z11223344, slice(0z001122334455, 1, -1))
938  assert_equal(0z11, slice(0z001122334455, 1, -4))
939  assert_equal(0z, slice(0z001122334455, 1, -5))
940  assert_equal(0z, slice(0z001122334455, 1, -6))
941enddef
942
943def Test_spellsuggest()
944  if !has('spell')
945    MissingFeature 'spell'
946  else
947    spellsuggest('marrch', 1, true)->assert_equal(['March'])
948  endif
949enddef
950
951def Test_sort_return_type()
952  var res: list<number>
953  res = [1, 2, 3]->sort()
954enddef
955
956def Test_sort_argument()
957  var lines =<< trim END
958    var res = ['b', 'a', 'c']->sort('i')
959    res->assert_equal(['a', 'b', 'c'])
960
961    def Compare(a: number, b: number): number
962      return a - b
963    enddef
964    var l = [3, 6, 7, 1, 8, 2, 4, 5]
965    sort(l, Compare)
966    assert_equal([1, 2, 3, 4, 5, 6, 7, 8], l)
967  END
968  CheckDefAndScriptSuccess(lines)
969enddef
970
971def Test_split()
972  split('  aa  bb  ', '\W\+', true)->assert_equal(['', 'aa', 'bb', ''])
973enddef
974
975def Run_str2float()
976  if !has('float')
977    MissingFeature 'float'
978  endif
979    str2float("1.00")->assert_equal(1.00)
980    str2float("2e-2")->assert_equal(0.02)
981
982    CheckDefFailure(['echo str2float(123)'], 'E1013:')
983    CheckScriptFailure(['vim9script', 'echo str2float(123)'], 'E1024:')
984  endif
985enddef
986
987def Test_str2nr()
988  str2nr("1'000'000", 10, true)->assert_equal(1000000)
989
990  CheckDefFailure(['echo str2nr(123)'], 'E1013:')
991  CheckScriptFailure(['vim9script', 'echo str2nr(123)'], 'E1024:')
992  CheckDefFailure(['echo str2nr("123", "x")'], 'E1013:')
993  CheckScriptFailure(['vim9script', 'echo str2nr("123", "x")'], 'E1030:')
994  CheckDefFailure(['echo str2nr("123", 10, "x")'], 'E1013:')
995  CheckScriptFailure(['vim9script', 'echo str2nr("123", 10, "x")'], 'E1135:')
996enddef
997
998def Test_strchars()
999  strchars("A\u20dd", true)->assert_equal(1)
1000enddef
1001
1002def Test_submatch()
1003  var pat = 'A\(.\)\(.\)\(.\)\(.\)\(.\)\(.\)\(.\)\(.\)\(.\)'
1004  var Rep = () => range(10)->mapnew((_, v) => submatch(v, true))->string()
1005  var actual = substitute('A123456789', pat, Rep, '')
1006  var expected = "[['A123456789'], ['1'], ['2'], ['3'], ['4'], ['5'], ['6'], ['7'], ['8'], ['9']]"
1007  actual->assert_equal(expected)
1008enddef
1009
1010def Test_synID()
1011  new
1012  setline(1, "text")
1013  synID(1, 1, true)->assert_equal(0)
1014  bwipe!
1015enddef
1016
1017def Test_term_gettty()
1018  if !has('terminal')
1019    MissingFeature 'terminal'
1020  else
1021    var buf = Run_shell_in_terminal({})
1022    term_gettty(buf, true)->assert_notequal('')
1023    StopShellInTerminal(buf)
1024  endif
1025enddef
1026
1027def Test_term_start()
1028  if !has('terminal')
1029    MissingFeature 'terminal'
1030  else
1031    botright new
1032    var winnr = winnr()
1033    term_start(&shell, {curwin: true})
1034    winnr()->assert_equal(winnr)
1035    bwipe!
1036  endif
1037enddef
1038
1039def Test_timer_paused()
1040  var id = timer_start(50, () => 0)
1041  timer_pause(id, true)
1042  var info = timer_info(id)
1043  info[0]['paused']->assert_equal(1)
1044  timer_stop(id)
1045enddef
1046
1047def Test_win_execute()
1048  assert_equal("\n" .. winnr(), win_execute(win_getid(), 'echo winnr()'))
1049  assert_equal('', win_execute(342343, 'echo winnr()'))
1050enddef
1051
1052def Test_win_splitmove()
1053  split
1054  win_splitmove(1, 2, {vertical: true, rightbelow: true})
1055  close
1056enddef
1057
1058def Test_winrestcmd()
1059  split
1060  var cmd = winrestcmd()
1061  wincmd _
1062  exe cmd
1063  assert_equal(cmd, winrestcmd())
1064  close
1065enddef
1066
1067def Test_winsaveview()
1068  var view: dict<number> = winsaveview()
1069
1070  var lines =<< trim END
1071      var view: list<number> = winsaveview()
1072  END
1073  CheckDefAndScriptFailure(lines, 'E1012: Type mismatch; expected list<number> but got dict<number>', 1)
1074enddef
1075
1076
1077
1078
1079" vim: ts=8 sw=2 sts=2 expandtab tw=80 fdm=marker
1080